mirror of
https://github.com/wiidev/usbloadergx.git
synced 2024-11-22 03:09:15 +01:00
* Unified formating of source files.
This commit is contained in:
parent
05825a3231
commit
772859ecbc
@ -2,8 +2,8 @@
|
|||||||
<app version="1">
|
<app version="1">
|
||||||
<name> USB Loader GX</name>
|
<name> USB Loader GX</name>
|
||||||
<coder>USB Loader GX Team</coder>
|
<coder>USB Loader GX Team</coder>
|
||||||
<version>1.0 r961</version>
|
<version>1.0 r962</version>
|
||||||
<release_date>201009222057</release_date>
|
<release_date>201009232353</release_date>
|
||||||
<short_description>Loads games from USB-devices</short_description>
|
<short_description>Loads games from USB-devices</short_description>
|
||||||
<long_description>USB Loader GX is a libwiigui based USB iso loader with a wii-like GUI. You can install games to your HDDs and boot them with shorter loading times.
|
<long_description>USB Loader GX is a libwiigui based USB iso loader with a wii-like GUI. You can install games to your HDDs and boot them with shorter loading times.
|
||||||
The interactive GUI is completely controllable with WiiMote, Classic Controller or GC Controller.
|
The interactive GUI is completely controllable with WiiMote, Classic Controller or GC Controller.
|
||||||
|
@ -27,55 +27,52 @@
|
|||||||
#include "filelist.h"
|
#include "filelist.h"
|
||||||
|
|
||||||
FreeTypeGX * fontSystem = NULL;
|
FreeTypeGX * fontSystem = NULL;
|
||||||
static FT_Byte * MainFont = ( FT_Byte * ) font_ttf;
|
static FT_Byte * MainFont = (FT_Byte *) font_ttf;
|
||||||
static u32 MainFontSize = font_ttf_size;
|
static u32 MainFontSize = font_ttf_size;
|
||||||
|
|
||||||
void ClearFontData()
|
void ClearFontData()
|
||||||
{
|
{
|
||||||
if ( fontSystem )
|
if (fontSystem) delete fontSystem;
|
||||||
delete fontSystem;
|
|
||||||
fontSystem = NULL;
|
fontSystem = NULL;
|
||||||
|
|
||||||
if ( MainFont != ( FT_Byte * ) font_ttf )
|
if (MainFont != (FT_Byte *) font_ttf)
|
||||||
{
|
{
|
||||||
if ( MainFont != NULL )
|
if (MainFont != NULL) delete[] MainFont;
|
||||||
delete [] MainFont;
|
MainFont = (FT_Byte *) font_ttf;
|
||||||
MainFont = ( FT_Byte * ) font_ttf;
|
|
||||||
MainFontSize = font_ttf_size;
|
MainFontSize = font_ttf_size;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool SetupDefaultFont( const char *path )
|
bool SetupDefaultFont(const char *path)
|
||||||
{
|
{
|
||||||
bool result = false;
|
bool result = false;
|
||||||
FILE *pfile = NULL;
|
FILE *pfile = NULL;
|
||||||
|
|
||||||
ClearFontData();
|
ClearFontData();
|
||||||
|
|
||||||
if ( path )
|
if (path) pfile = fopen(path, "rb");
|
||||||
pfile = fopen( path, "rb" );
|
|
||||||
|
|
||||||
if ( pfile )
|
if (pfile)
|
||||||
{
|
{
|
||||||
fseek( pfile, 0, SEEK_END );
|
fseek(pfile, 0, SEEK_END);
|
||||||
MainFontSize = ftell( pfile );
|
MainFontSize = ftell(pfile);
|
||||||
rewind( pfile );
|
rewind(pfile);
|
||||||
|
|
||||||
MainFont = new ( std::nothrow ) FT_Byte[MainFontSize];
|
MainFont = new (std::nothrow) FT_Byte[MainFontSize];
|
||||||
if ( !MainFont )
|
if (!MainFont)
|
||||||
{
|
{
|
||||||
MainFont = ( FT_Byte * ) font_ttf;
|
MainFont = (FT_Byte *) font_ttf;
|
||||||
MainFontSize = font_ttf_size;
|
MainFontSize = font_ttf_size;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
fread( MainFont, 1, MainFontSize, pfile );
|
fread(MainFont, 1, MainFontSize, pfile);
|
||||||
result = true;
|
result = true;
|
||||||
}
|
}
|
||||||
fclose( pfile );
|
fclose(pfile);
|
||||||
}
|
}
|
||||||
|
|
||||||
fontSystem = new FreeTypeGX( MainFont, MainFontSize );
|
fontSystem = new FreeTypeGX(MainFont, MainFontSize);
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -26,7 +26,7 @@
|
|||||||
#ifndef FONTSYSTEM_H_
|
#ifndef FONTSYSTEM_H_
|
||||||
#define FONTSYSTEM_H_
|
#define FONTSYSTEM_H_
|
||||||
|
|
||||||
bool SetupDefaultFont( const char *path );
|
bool SetupDefaultFont(const char *path);
|
||||||
void ClearFontData();
|
void ClearFontData();
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -34,24 +34,23 @@ using namespace std;
|
|||||||
* @return Wide character representation of supplied character string.
|
* @return Wide character representation of supplied character string.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
wchar_t* charToWideChar( const char* strChar )
|
wchar_t* charToWideChar(const char* strChar)
|
||||||
{
|
{
|
||||||
if ( !strChar )
|
if (!strChar) return NULL;
|
||||||
return NULL;
|
|
||||||
|
|
||||||
wchar_t *strWChar = new ( std::nothrow ) wchar_t[strlen( strChar ) + 1];
|
wchar_t *strWChar = new (std::nothrow) wchar_t[strlen(strChar) + 1];
|
||||||
if ( !strWChar )
|
if (!strWChar) return NULL;
|
||||||
return NULL;
|
|
||||||
|
|
||||||
int bt = mbstowcs( strWChar, strChar, strlen( strChar ) );
|
int bt = mbstowcs(strWChar, strChar, strlen(strChar));
|
||||||
if ( bt > 0 )
|
if (bt > 0)
|
||||||
{
|
{
|
||||||
strWChar[bt] = 0;
|
strWChar[bt] = 0;
|
||||||
return strWChar;
|
return strWChar;
|
||||||
}
|
}
|
||||||
|
|
||||||
wchar_t *tempDest = strWChar;
|
wchar_t *tempDest = strWChar;
|
||||||
while ( ( *tempDest++ = *strChar++ ) );
|
while ((*tempDest++ = *strChar++))
|
||||||
|
;
|
||||||
|
|
||||||
return strWChar;
|
return strWChar;
|
||||||
}
|
}
|
||||||
@ -59,14 +58,14 @@ wchar_t* charToWideChar( const char* strChar )
|
|||||||
/**
|
/**
|
||||||
* Default constructor for the FreeTypeGX class for WiiXplorer.
|
* Default constructor for the FreeTypeGX class for WiiXplorer.
|
||||||
*/
|
*/
|
||||||
FreeTypeGX::FreeTypeGX( const uint8_t* fontBuffer, FT_Long bufferSize )
|
FreeTypeGX::FreeTypeGX(const uint8_t* fontBuffer, FT_Long bufferSize)
|
||||||
{
|
{
|
||||||
ftPointSize = 0;
|
ftPointSize = 0;
|
||||||
|
|
||||||
FT_Init_FreeType( &ftLibrary );
|
FT_Init_FreeType(&ftLibrary);
|
||||||
FT_New_Memory_Face( ftLibrary, ( FT_Byte * )fontBuffer, bufferSize, 0, &ftFace );
|
FT_New_Memory_Face(ftLibrary, (FT_Byte *) fontBuffer, bufferSize, 0, &ftFace);
|
||||||
|
|
||||||
setVertexFormat( GX_VTXFMT1 );
|
setVertexFormat(GX_VTXFMT1);
|
||||||
ftKerningEnabled = FT_HAS_KERNING( ftFace );
|
ftKerningEnabled = FT_HAS_KERNING( ftFace );
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -76,8 +75,8 @@ FreeTypeGX::FreeTypeGX( const uint8_t* fontBuffer, FT_Long bufferSize )
|
|||||||
FreeTypeGX::~FreeTypeGX()
|
FreeTypeGX::~FreeTypeGX()
|
||||||
{
|
{
|
||||||
unloadFont();
|
unloadFont();
|
||||||
FT_Done_Face( ftFace );
|
FT_Done_Face(ftFace);
|
||||||
FT_Done_FreeType( ftLibrary );
|
FT_Done_FreeType(ftLibrary);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -88,13 +87,13 @@ FreeTypeGX::~FreeTypeGX()
|
|||||||
* vertex format index is modified.
|
* vertex format index is modified.
|
||||||
*
|
*
|
||||||
* @param vertexIndex Vertex format index (GX_VTXFMT*) of the glyph textures as defined by the libogc gx.h header file.
|
* @param vertexIndex Vertex format index (GX_VTXFMT*) of the glyph textures as defined by the libogc gx.h header file.
|
||||||
*/
|
*/
|
||||||
void FreeTypeGX::setVertexFormat( uint8_t vertexInd )
|
void FreeTypeGX::setVertexFormat(uint8_t vertexInd)
|
||||||
{
|
{
|
||||||
vertexIndex = vertexInd;
|
vertexIndex = vertexInd;
|
||||||
GX_SetVtxAttrFmt( vertexIndex, GX_VA_POS, GX_POS_XYZ, GX_S16, 0 );
|
GX_SetVtxAttrFmt(vertexIndex, GX_VA_POS, GX_POS_XYZ, GX_S16, 0);
|
||||||
GX_SetVtxAttrFmt( vertexIndex, GX_VA_TEX0, GX_TEX_ST, GX_F32, 0 );
|
GX_SetVtxAttrFmt(vertexIndex, GX_VA_TEX0, GX_TEX_ST, GX_F32, 0);
|
||||||
GX_SetVtxAttrFmt( vertexIndex, GX_VA_CLR0, GX_CLR_RGBA, GX_RGBA8, 0 );
|
GX_SetVtxAttrFmt(vertexIndex, GX_VA_CLR0, GX_CLR_RGBA, GX_RGBA8, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -104,16 +103,15 @@ void FreeTypeGX::setVertexFormat( uint8_t vertexInd )
|
|||||||
*/
|
*/
|
||||||
void FreeTypeGX::unloadFont()
|
void FreeTypeGX::unloadFont()
|
||||||
{
|
{
|
||||||
if ( this->fontData.size() == 0 )
|
if (this->fontData.size() == 0) return;
|
||||||
return;
|
|
||||||
|
|
||||||
map<int16_t, map<wchar_t, ftgxCharData> >::iterator itr;
|
map<int16_t, map<wchar_t, ftgxCharData> >::iterator itr;
|
||||||
map<wchar_t, ftgxCharData>::iterator itr2;
|
map<wchar_t, ftgxCharData>::iterator itr2;
|
||||||
|
|
||||||
for ( itr = fontData.begin(); itr != fontData.end(); itr++ )
|
for (itr = fontData.begin(); itr != fontData.end(); itr++)
|
||||||
{
|
{
|
||||||
for ( itr2 = itr->second.begin(); itr2 != itr->second.end(); itr2++ )
|
for (itr2 = itr->second.begin(); itr2 != itr->second.end(); itr2++)
|
||||||
free( itr2->second.glyphDataTexture );
|
free(itr2->second.glyphDataTexture);
|
||||||
|
|
||||||
itr->second.clear();
|
itr->second.clear();
|
||||||
}
|
}
|
||||||
@ -131,17 +129,17 @@ void FreeTypeGX::unloadFont()
|
|||||||
* @param charCode The requested glyph's character code.
|
* @param charCode The requested glyph's character code.
|
||||||
* @return A pointer to the allocated font structure.
|
* @return A pointer to the allocated font structure.
|
||||||
*/
|
*/
|
||||||
ftgxCharData * FreeTypeGX::cacheGlyphData( wchar_t charCode, int16_t pixelSize )
|
ftgxCharData * FreeTypeGX::cacheGlyphData(wchar_t charCode, int16_t pixelSize)
|
||||||
{
|
{
|
||||||
map<int16_t, map<wchar_t, ftgxCharData> >::iterator itr;
|
map<int16_t, map<wchar_t, ftgxCharData> >::iterator itr;
|
||||||
map<wchar_t, ftgxCharData>::iterator itr2;
|
map<wchar_t, ftgxCharData>::iterator itr2;
|
||||||
|
|
||||||
itr = fontData.find( pixelSize );
|
itr = fontData.find(pixelSize);
|
||||||
if ( itr != fontData.end() )
|
if (itr != fontData.end())
|
||||||
{
|
{
|
||||||
itr2 = itr->second.find( charCode );
|
itr2 = itr->second.find(charCode);
|
||||||
|
|
||||||
if ( itr2 != itr->second.end() )
|
if (itr2 != itr->second.end())
|
||||||
{
|
{
|
||||||
return &itr2->second;
|
return &itr2->second;
|
||||||
}
|
}
|
||||||
@ -150,43 +148,43 @@ ftgxCharData * FreeTypeGX::cacheGlyphData( wchar_t charCode, int16_t pixelSize )
|
|||||||
FT_UInt gIndex;
|
FT_UInt gIndex;
|
||||||
uint16_t textureWidth = 0, textureHeight = 0;
|
uint16_t textureWidth = 0, textureHeight = 0;
|
||||||
|
|
||||||
if ( ftPointSize != pixelSize )
|
if (ftPointSize != pixelSize)
|
||||||
{
|
{
|
||||||
ftPointSize = pixelSize;
|
ftPointSize = pixelSize;
|
||||||
FT_Set_Pixel_Sizes( ftFace, 0, ftPointSize );
|
FT_Set_Pixel_Sizes(ftFace, 0, ftPointSize);
|
||||||
|
|
||||||
//!Cache ascender and decender as well
|
//!Cache ascender and decender as well
|
||||||
map<int16_t, ftgxDataOffset>::iterator itrAlign = ftgxAlign.find( ftPointSize );
|
map<int16_t, ftgxDataOffset>::iterator itrAlign = ftgxAlign.find(ftPointSize);
|
||||||
if ( itrAlign == ftgxAlign.end() )
|
if (itrAlign == ftgxAlign.end())
|
||||||
{
|
{
|
||||||
ftgxAlign[ftPointSize].ascender = ( int16_t ) ftFace->size->metrics.ascender >> 6;
|
ftgxAlign[ftPointSize].ascender = (int16_t) ftFace->size->metrics.ascender >> 6;
|
||||||
ftgxAlign[ftPointSize].descender = ( int16_t ) ftFace->size->metrics.descender >> 6;
|
ftgxAlign[ftPointSize].descender = (int16_t) ftFace->size->metrics.descender >> 6;
|
||||||
ftgxAlign[ftPointSize].max = 0;
|
ftgxAlign[ftPointSize].max = 0;
|
||||||
ftgxAlign[ftPointSize].min = 0;
|
ftgxAlign[ftPointSize].min = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
gIndex = FT_Get_Char_Index( ftFace, ( FT_ULong ) charCode );
|
gIndex = FT_Get_Char_Index(ftFace, (FT_ULong) charCode);
|
||||||
if ( gIndex != 0 && FT_Load_Glyph( ftFace, gIndex, FT_LOAD_DEFAULT | FT_LOAD_RENDER ) == 0 )
|
if (gIndex != 0 && FT_Load_Glyph(ftFace, gIndex, FT_LOAD_DEFAULT | FT_LOAD_RENDER) == 0)
|
||||||
{
|
{
|
||||||
if ( ftFace->glyph->format == FT_GLYPH_FORMAT_BITMAP )
|
if (ftFace->glyph->format == FT_GLYPH_FORMAT_BITMAP)
|
||||||
{
|
{
|
||||||
FT_Bitmap *glyphBitmap = &ftFace->glyph->bitmap;
|
FT_Bitmap *glyphBitmap = &ftFace->glyph->bitmap;
|
||||||
|
|
||||||
textureWidth = glyphBitmap->width + ( 4 - glyphBitmap->width % 4 ) % 4;
|
textureWidth = glyphBitmap->width + (4 - glyphBitmap->width % 4) % 4;
|
||||||
textureHeight = glyphBitmap->rows + ( 4 - glyphBitmap->rows % 4 ) % 4;
|
textureHeight = glyphBitmap->rows + (4 - glyphBitmap->rows % 4) % 4;
|
||||||
|
|
||||||
fontData[pixelSize][charCode].renderOffsetX = ( int16_t ) ftFace->glyph->bitmap_left;
|
fontData[pixelSize][charCode].renderOffsetX = (int16_t) ftFace->glyph->bitmap_left;
|
||||||
fontData[pixelSize][charCode].glyphAdvanceX = ( uint16_t ) ( ftFace->glyph->advance.x >> 6 );
|
fontData[pixelSize][charCode].glyphAdvanceX = (uint16_t) (ftFace->glyph->advance.x >> 6);
|
||||||
fontData[pixelSize][charCode].glyphIndex = ( uint32_t ) gIndex;
|
fontData[pixelSize][charCode].glyphIndex = (uint32_t) gIndex;
|
||||||
fontData[pixelSize][charCode].textureWidth = ( uint16_t ) textureWidth;
|
fontData[pixelSize][charCode].textureWidth = (uint16_t) textureWidth;
|
||||||
fontData[pixelSize][charCode].textureHeight = ( uint16_t ) textureHeight;
|
fontData[pixelSize][charCode].textureHeight = (uint16_t) textureHeight;
|
||||||
fontData[pixelSize][charCode].renderOffsetY = ( int16_t ) ftFace->glyph->bitmap_top;
|
fontData[pixelSize][charCode].renderOffsetY = (int16_t) ftFace->glyph->bitmap_top;
|
||||||
fontData[pixelSize][charCode].renderOffsetMax = ( int16_t ) ftFace->glyph->bitmap_top;
|
fontData[pixelSize][charCode].renderOffsetMax = (int16_t) ftFace->glyph->bitmap_top;
|
||||||
fontData[pixelSize][charCode].renderOffsetMin = ( int16_t ) glyphBitmap->rows - ftFace->glyph->bitmap_top;
|
fontData[pixelSize][charCode].renderOffsetMin = (int16_t) glyphBitmap->rows - ftFace->glyph->bitmap_top;
|
||||||
fontData[pixelSize][charCode].glyphDataTexture = NULL;
|
fontData[pixelSize][charCode].glyphDataTexture = NULL;
|
||||||
|
|
||||||
loadGlyphData( glyphBitmap, &fontData[pixelSize][charCode] );
|
loadGlyphData(glyphBitmap, &fontData[pixelSize][charCode]);
|
||||||
|
|
||||||
return &fontData[pixelSize][charCode];
|
return &fontData[pixelSize][charCode];
|
||||||
}
|
}
|
||||||
@ -200,19 +198,18 @@ ftgxCharData * FreeTypeGX::cacheGlyphData( wchar_t charCode, int16_t pixelSize )
|
|||||||
* This routine locates each character in the configured font face and renders the glyph's bitmap.
|
* This routine locates each character in the configured font face and renders the glyph's bitmap.
|
||||||
* Each bitmap and relevant information is loaded into its own quickly addressible structure within an instance-specific map.
|
* Each bitmap and relevant information is loaded into its own quickly addressible structure within an instance-specific map.
|
||||||
*/
|
*/
|
||||||
uint16_t FreeTypeGX::cacheGlyphDataComplete( int16_t pixelSize )
|
uint16_t FreeTypeGX::cacheGlyphDataComplete(int16_t pixelSize)
|
||||||
{
|
{
|
||||||
uint32_t i = 0;
|
uint32_t i = 0;
|
||||||
FT_UInt gIndex;
|
FT_UInt gIndex;
|
||||||
|
|
||||||
FT_ULong charCode = FT_Get_First_Char( ftFace, &gIndex );
|
FT_ULong charCode = FT_Get_First_Char(ftFace, &gIndex);
|
||||||
while ( gIndex != 0 )
|
while (gIndex != 0)
|
||||||
{
|
{
|
||||||
if ( cacheGlyphData( charCode, pixelSize ) != NULL )
|
if (cacheGlyphData(charCode, pixelSize) != NULL) ++i;
|
||||||
++i;
|
charCode = FT_Get_Next_Char(ftFace, charCode, &gIndex);
|
||||||
charCode = FT_Get_Next_Char( ftFace, charCode, &gIndex );
|
|
||||||
}
|
}
|
||||||
return ( uint16_t )( i );
|
return (uint16_t) (i);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -227,32 +224,32 @@ uint16_t FreeTypeGX::cacheGlyphDataComplete( int16_t pixelSize )
|
|||||||
*
|
*
|
||||||
* Optimized for RGBA8 use by Dimok.
|
* Optimized for RGBA8 use by Dimok.
|
||||||
*/
|
*/
|
||||||
void FreeTypeGX::loadGlyphData( FT_Bitmap *bmp, ftgxCharData *charData )
|
void FreeTypeGX::loadGlyphData(FT_Bitmap *bmp, ftgxCharData *charData)
|
||||||
{
|
{
|
||||||
int length = ( ( ( ( charData->textureWidth + 3 ) >> 2 ) * ( ( charData->textureHeight + 3 ) >> 2 ) * 32 * 2 + 31 ) & ~31 );
|
int length = ((((charData->textureWidth + 3) >> 2) * ((charData->textureHeight + 3) >> 2) * 32 * 2 + 31) & ~31);
|
||||||
|
|
||||||
uint8_t * glyphData = ( uint8_t * ) memalign( 32, length );
|
uint8_t * glyphData = (uint8_t *) memalign(32, length);
|
||||||
if ( !glyphData )
|
if (!glyphData) return;
|
||||||
return;
|
|
||||||
|
|
||||||
memset( glyphData, 0x00, length );
|
memset(glyphData, 0x00, length);
|
||||||
|
|
||||||
uint8_t *src = ( uint8_t * )bmp->buffer;
|
uint8_t *src = (uint8_t *) bmp->buffer;
|
||||||
uint32_t offset;
|
uint32_t offset;
|
||||||
|
|
||||||
for ( int imagePosY = 0; imagePosY < bmp->rows; ++imagePosY )
|
for (int imagePosY = 0; imagePosY < bmp->rows; ++imagePosY)
|
||||||
{
|
{
|
||||||
for ( int imagePosX = 0; imagePosX < bmp->width; ++imagePosX )
|
for (int imagePosX = 0; imagePosX < bmp->width; ++imagePosX)
|
||||||
{
|
{
|
||||||
offset = ( ( ( ( imagePosY >> 2 ) * ( charData->textureWidth >> 2 ) + ( imagePosX >> 2 ) ) << 5 ) + ( ( imagePosY & 3 ) << 2 ) + ( imagePosX & 3 ) ) << 1;
|
offset = ((((imagePosY >> 2) * (charData->textureWidth >> 2) + (imagePosX >> 2)) << 5) + ((imagePosY & 3)
|
||||||
|
<< 2) + (imagePosX & 3)) << 1;
|
||||||
glyphData[offset] = *src;
|
glyphData[offset] = *src;
|
||||||
glyphData[offset+1] = *src;
|
glyphData[offset + 1] = *src;
|
||||||
glyphData[offset+32] = *src;
|
glyphData[offset + 32] = *src;
|
||||||
glyphData[offset+33] = *src;
|
glyphData[offset + 33] = *src;
|
||||||
++src;
|
++src;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DCFlushRange( glyphData, length );
|
DCFlushRange(glyphData, length);
|
||||||
|
|
||||||
charData->glyphDataTexture = glyphData;
|
charData->glyphDataTexture = glyphData;
|
||||||
}
|
}
|
||||||
@ -265,14 +262,13 @@ void FreeTypeGX::loadGlyphData( FT_Bitmap *bmp, ftgxCharData *charData )
|
|||||||
* @param width Current pixel width of the string.
|
* @param width Current pixel width of the string.
|
||||||
* @param format Positional format of the string.
|
* @param format Positional format of the string.
|
||||||
*/
|
*/
|
||||||
int16_t FreeTypeGX::getStyleOffsetWidth( uint16_t width, uint16_t format )
|
int16_t FreeTypeGX::getStyleOffsetWidth(uint16_t width, uint16_t format)
|
||||||
{
|
{
|
||||||
if ( format & FTGX_JUSTIFY_LEFT )
|
if (format & FTGX_JUSTIFY_LEFT)
|
||||||
return 0;
|
return 0;
|
||||||
else if ( format & FTGX_JUSTIFY_CENTER )
|
else if (format & FTGX_JUSTIFY_CENTER)
|
||||||
return -( width >> 1 );
|
return -(width >> 1);
|
||||||
else if ( format & FTGX_JUSTIFY_RIGHT )
|
else if (format & FTGX_JUSTIFY_RIGHT) return -width;
|
||||||
return -width;
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -284,20 +280,19 @@ int16_t FreeTypeGX::getStyleOffsetWidth( uint16_t width, uint16_t format )
|
|||||||
* @param offset Current pixel offset data of the string.
|
* @param offset Current pixel offset data of the string.
|
||||||
* @param format Positional format of the string.
|
* @param format Positional format of the string.
|
||||||
*/
|
*/
|
||||||
int16_t FreeTypeGX::getStyleOffsetHeight( int16_t format, uint16_t pixelSize )
|
int16_t FreeTypeGX::getStyleOffsetHeight(int16_t format, uint16_t pixelSize)
|
||||||
{
|
{
|
||||||
map<int16_t, ftgxDataOffset>::iterator itrAlign = ftgxAlign.find( pixelSize );
|
map<int16_t, ftgxDataOffset>::iterator itrAlign = ftgxAlign.find(pixelSize);
|
||||||
if ( itrAlign == ftgxAlign.end() )
|
if (itrAlign == ftgxAlign.end()) return 0;
|
||||||
return 0;
|
|
||||||
|
|
||||||
switch ( format & FTGX_ALIGN_MASK )
|
switch (format & FTGX_ALIGN_MASK)
|
||||||
{
|
{
|
||||||
case FTGX_ALIGN_TOP:
|
case FTGX_ALIGN_TOP:
|
||||||
return itrAlign->second.ascender;
|
return itrAlign->second.ascender;
|
||||||
|
|
||||||
case FTGX_ALIGN_MIDDLE:
|
case FTGX_ALIGN_MIDDLE:
|
||||||
default:
|
default:
|
||||||
return ( itrAlign->second.ascender + itrAlign->second.descender + 1 ) >> 1;
|
return (itrAlign->second.ascender + itrAlign->second.descender + 1) >> 1;
|
||||||
|
|
||||||
case FTGX_ALIGN_BOTTOM:
|
case FTGX_ALIGN_BOTTOM:
|
||||||
return itrAlign->second.descender;
|
return itrAlign->second.descender;
|
||||||
@ -309,7 +304,7 @@ int16_t FreeTypeGX::getStyleOffsetHeight( int16_t format, uint16_t pixelSize )
|
|||||||
return itrAlign->second.max;
|
return itrAlign->second.max;
|
||||||
|
|
||||||
case FTGX_ALIGN_GLYPH_MIDDLE:
|
case FTGX_ALIGN_GLYPH_MIDDLE:
|
||||||
return ( itrAlign->second.max + itrAlign->second.min + 1 ) >> 1;
|
return (itrAlign->second.max + itrAlign->second.min + 1) >> 1;
|
||||||
|
|
||||||
case FTGX_ALIGN_GLYPH_BOTTOM:
|
case FTGX_ALIGN_GLYPH_BOTTOM:
|
||||||
return itrAlign->second.min;
|
return itrAlign->second.min;
|
||||||
@ -330,45 +325,47 @@ int16_t FreeTypeGX::getStyleOffsetHeight( int16_t format, uint16_t pixelSize )
|
|||||||
* @param textStyle Flags which specify any styling which should be applied to the rendered string.
|
* @param textStyle Flags which specify any styling which should be applied to the rendered string.
|
||||||
* @return The number of characters printed.
|
* @return The number of characters printed.
|
||||||
*/
|
*/
|
||||||
uint16_t FreeTypeGX::drawText( int16_t x, int16_t y, int16_t z, const wchar_t *text, int16_t pixelSize, GXColor color, uint16_t textStyle, uint16_t textWidth, uint16_t widthLimit )
|
uint16_t FreeTypeGX::drawText(int16_t x, int16_t y, int16_t z, const wchar_t *text, int16_t pixelSize, GXColor color,
|
||||||
|
uint16_t textStyle, uint16_t textWidth, uint16_t widthLimit)
|
||||||
{
|
{
|
||||||
if ( !text )
|
if (!text) return 0;
|
||||||
return 0;
|
|
||||||
|
|
||||||
uint16_t fullTextWidth = textWidth > 0 ? textWidth : getWidth( text, pixelSize );
|
uint16_t fullTextWidth = textWidth > 0 ? textWidth : getWidth(text, pixelSize);
|
||||||
uint16_t x_pos = x, printed = 0;
|
uint16_t x_pos = x, printed = 0;
|
||||||
uint16_t x_offset = 0, y_offset = 0;
|
uint16_t x_offset = 0, y_offset = 0;
|
||||||
GXTexObj glyphTexture;
|
GXTexObj glyphTexture;
|
||||||
FT_Vector pairDelta;
|
FT_Vector pairDelta;
|
||||||
|
|
||||||
if ( textStyle & FTGX_JUSTIFY_MASK )
|
if (textStyle & FTGX_JUSTIFY_MASK)
|
||||||
{
|
{
|
||||||
x_offset = getStyleOffsetWidth( fullTextWidth, textStyle );
|
x_offset = getStyleOffsetWidth(fullTextWidth, textStyle);
|
||||||
}
|
}
|
||||||
if ( textStyle & FTGX_ALIGN_MASK )
|
if (textStyle & FTGX_ALIGN_MASK)
|
||||||
{
|
{
|
||||||
y_offset = getStyleOffsetHeight( textStyle, pixelSize );
|
y_offset = getStyleOffsetHeight(textStyle, pixelSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
int i = 0;
|
int i = 0;
|
||||||
|
|
||||||
while ( text[i] )
|
while (text[i])
|
||||||
{
|
{
|
||||||
if ( widthLimit > 0 && ( x_pos - x ) > widthLimit )
|
if (widthLimit > 0 && (x_pos - x) > widthLimit) break;
|
||||||
break;
|
|
||||||
|
|
||||||
ftgxCharData* glyphData = cacheGlyphData( text[i], pixelSize );
|
ftgxCharData* glyphData = cacheGlyphData(text[i], pixelSize);
|
||||||
|
|
||||||
if ( glyphData != NULL )
|
if (glyphData != NULL)
|
||||||
{
|
{
|
||||||
if ( ftKerningEnabled && i > 0 )
|
if (ftKerningEnabled && i > 0)
|
||||||
{
|
{
|
||||||
FT_Get_Kerning( ftFace, fontData[pixelSize][text[i - 1]].glyphIndex, glyphData->glyphIndex, FT_KERNING_DEFAULT, &pairDelta );
|
FT_Get_Kerning(ftFace, fontData[pixelSize][text[i - 1]].glyphIndex, glyphData->glyphIndex,
|
||||||
|
FT_KERNING_DEFAULT, &pairDelta);
|
||||||
x_pos += pairDelta.x >> 6;
|
x_pos += pairDelta.x >> 6;
|
||||||
}
|
}
|
||||||
|
|
||||||
GX_InitTexObj( &glyphTexture, glyphData->glyphDataTexture, glyphData->textureWidth, glyphData->textureHeight, GX_TF_RGBA8, GX_CLAMP, GX_CLAMP, GX_FALSE );
|
GX_InitTexObj(&glyphTexture, glyphData->glyphDataTexture, glyphData->textureWidth,
|
||||||
copyTextureToFramebuffer( &glyphTexture, glyphData->textureWidth, glyphData->textureHeight, x_pos + glyphData->renderOffsetX + x_offset, y - glyphData->renderOffsetY + y_offset, z, color );
|
glyphData->textureHeight, GX_TF_RGBA8, GX_CLAMP, GX_CLAMP, GX_FALSE);
|
||||||
|
copyTextureToFramebuffer(&glyphTexture, glyphData->textureWidth, glyphData->textureHeight, x_pos
|
||||||
|
+ glyphData->renderOffsetX + x_offset, y - glyphData->renderOffsetY + y_offset, z, color);
|
||||||
|
|
||||||
x_pos += glyphData->glyphAdvanceX;
|
x_pos += glyphData->glyphAdvanceX;
|
||||||
++printed;
|
++printed;
|
||||||
@ -376,24 +373,25 @@ uint16_t FreeTypeGX::drawText( int16_t x, int16_t y, int16_t z, const wchar_t *t
|
|||||||
++i;
|
++i;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( textStyle & FTGX_STYLE_MASK )
|
if (textStyle & FTGX_STYLE_MASK)
|
||||||
{
|
{
|
||||||
getOffset( text, pixelSize, widthLimit );
|
getOffset(text, pixelSize, widthLimit);
|
||||||
drawTextFeature( x + x_offset, y + y_offset, z, pixelSize, fullTextWidth, &ftgxAlign[pixelSize], textStyle, color );
|
drawTextFeature(x + x_offset, y + y_offset, z, pixelSize, fullTextWidth, &ftgxAlign[pixelSize], textStyle,
|
||||||
|
color);
|
||||||
}
|
}
|
||||||
|
|
||||||
return printed;
|
return printed;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FreeTypeGX::drawTextFeature( int16_t x, int16_t y, int16_t z, int16_t pixelSize, uint16_t width, ftgxDataOffset *offsetData, uint16_t format, GXColor color )
|
void FreeTypeGX::drawTextFeature(int16_t x, int16_t y, int16_t z, int16_t pixelSize, uint16_t width,
|
||||||
|
ftgxDataOffset *offsetData, uint16_t format, GXColor color)
|
||||||
{
|
{
|
||||||
uint16_t featureHeight = pixelSize >> 4 > 0 ? pixelSize >> 4 : 1;
|
uint16_t featureHeight = pixelSize >> 4 > 0 ? pixelSize >> 4 : 1;
|
||||||
|
|
||||||
if ( format & FTGX_STYLE_UNDERLINE )
|
if (format & FTGX_STYLE_UNDERLINE) this->copyFeatureToFramebuffer(width, featureHeight, x, y + 1, z, color);
|
||||||
this->copyFeatureToFramebuffer( width, featureHeight, x, y + 1, z, color );
|
|
||||||
|
|
||||||
if ( format & FTGX_STYLE_STRIKE )
|
if (format & FTGX_STYLE_STRIKE) this->copyFeatureToFramebuffer(width, featureHeight, x, y
|
||||||
this->copyFeatureToFramebuffer( width, featureHeight, x, y - ( ( offsetData->max ) >> 1 ), z, color );
|
- ((offsetData->max) >> 1), z, color);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -405,24 +403,24 @@ void FreeTypeGX::drawTextFeature( int16_t x, int16_t y, int16_t z, int16_t pixel
|
|||||||
* @param text NULL terminated string to calculate.
|
* @param text NULL terminated string to calculate.
|
||||||
* @return The width of the text string in pixels.
|
* @return The width of the text string in pixels.
|
||||||
*/
|
*/
|
||||||
uint16_t FreeTypeGX::getWidth( const wchar_t *text, int16_t pixelSize )
|
uint16_t FreeTypeGX::getWidth(const wchar_t *text, int16_t pixelSize)
|
||||||
{
|
{
|
||||||
if ( !text )
|
if (!text) return 0;
|
||||||
return 0;
|
|
||||||
|
|
||||||
uint16_t strWidth = 0;
|
uint16_t strWidth = 0;
|
||||||
FT_Vector pairDelta;
|
FT_Vector pairDelta;
|
||||||
|
|
||||||
int i = 0;
|
int i = 0;
|
||||||
while ( text[i] )
|
while (text[i])
|
||||||
{
|
{
|
||||||
ftgxCharData* glyphData = cacheGlyphData( text[i], pixelSize );
|
ftgxCharData* glyphData = cacheGlyphData(text[i], pixelSize);
|
||||||
|
|
||||||
if ( glyphData != NULL )
|
if (glyphData != NULL)
|
||||||
{
|
{
|
||||||
if ( ftKerningEnabled && ( i > 0 ) )
|
if (ftKerningEnabled && (i > 0))
|
||||||
{
|
{
|
||||||
FT_Get_Kerning( ftFace, fontData[pixelSize][text[i - 1]].glyphIndex, glyphData->glyphIndex, FT_KERNING_DEFAULT, &pairDelta );
|
FT_Get_Kerning(ftFace, fontData[pixelSize][text[i - 1]].glyphIndex, glyphData->glyphIndex,
|
||||||
|
FT_KERNING_DEFAULT, &pairDelta);
|
||||||
strWidth += pairDelta.x >> 6;
|
strWidth += pairDelta.x >> 6;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -435,18 +433,19 @@ uint16_t FreeTypeGX::getWidth( const wchar_t *text, int16_t pixelSize )
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Single char width
|
* Single char width
|
||||||
*/
|
*/
|
||||||
uint16_t FreeTypeGX::getCharWidth( const wchar_t wChar, int16_t pixelSize, const wchar_t prevChar )
|
uint16_t FreeTypeGX::getCharWidth(const wchar_t wChar, int16_t pixelSize, const wchar_t prevChar)
|
||||||
{
|
{
|
||||||
uint16_t strWidth = 0;
|
uint16_t strWidth = 0;
|
||||||
ftgxCharData * glyphData = cacheGlyphData( wChar, pixelSize );
|
ftgxCharData * glyphData = cacheGlyphData(wChar, pixelSize);
|
||||||
|
|
||||||
if ( glyphData != NULL )
|
if (glyphData != NULL)
|
||||||
{
|
{
|
||||||
if ( ftKerningEnabled && prevChar != 0x0000 )
|
if (ftKerningEnabled && prevChar != 0x0000)
|
||||||
{
|
{
|
||||||
FT_Vector pairDelta;
|
FT_Vector pairDelta;
|
||||||
FT_Get_Kerning( ftFace, fontData[pixelSize][prevChar].glyphIndex, glyphData->glyphIndex, FT_KERNING_DEFAULT, &pairDelta );
|
FT_Get_Kerning(ftFace, fontData[pixelSize][prevChar].glyphIndex, glyphData->glyphIndex, FT_KERNING_DEFAULT,
|
||||||
|
&pairDelta);
|
||||||
strWidth += pairDelta.x >> 6;
|
strWidth += pairDelta.x >> 6;
|
||||||
}
|
}
|
||||||
strWidth += glyphData->glyphAdvanceX;
|
strWidth += glyphData->glyphAdvanceX;
|
||||||
@ -464,9 +463,9 @@ uint16_t FreeTypeGX::getCharWidth( const wchar_t wChar, int16_t pixelSize, const
|
|||||||
* @param text NULL terminated string to calculate.
|
* @param text NULL terminated string to calculate.
|
||||||
* @return The height of the text string in pixels.
|
* @return The height of the text string in pixels.
|
||||||
*/
|
*/
|
||||||
uint16_t FreeTypeGX::getHeight( const wchar_t *text, int16_t pixelSize )
|
uint16_t FreeTypeGX::getHeight(const wchar_t *text, int16_t pixelSize)
|
||||||
{
|
{
|
||||||
getOffset( text, pixelSize );
|
getOffset(text, pixelSize);
|
||||||
|
|
||||||
return ftgxAlign[pixelSize].max - ftgxAlign[pixelSize].min;
|
return ftgxAlign[pixelSize].max - ftgxAlign[pixelSize].min;
|
||||||
}
|
}
|
||||||
@ -481,24 +480,22 @@ uint16_t FreeTypeGX::getHeight( const wchar_t *text, int16_t pixelSize )
|
|||||||
* @param offset returns the max and min values above and below the font origin line
|
* @param offset returns the max and min values above and below the font origin line
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
void FreeTypeGX::getOffset( const wchar_t *text, int16_t pixelSize, uint16_t widthLimit )
|
void FreeTypeGX::getOffset(const wchar_t *text, int16_t pixelSize, uint16_t widthLimit)
|
||||||
{
|
{
|
||||||
if ( ftgxAlign.find( pixelSize ) != ftgxAlign.end() )
|
if (ftgxAlign.find(pixelSize) != ftgxAlign.end()) return;
|
||||||
return;
|
|
||||||
|
|
||||||
int16_t strMax = 0, strMin = 9999;
|
int16_t strMax = 0, strMin = 9999;
|
||||||
uint16_t currWidth = 0;
|
uint16_t currWidth = 0;
|
||||||
|
|
||||||
int i = 0;
|
int i = 0;
|
||||||
|
|
||||||
while ( text[i] )
|
while (text[i])
|
||||||
{
|
{
|
||||||
if ( widthLimit > 0 && currWidth >= widthLimit )
|
if (widthLimit > 0 && currWidth >= widthLimit) break;
|
||||||
break;
|
|
||||||
|
|
||||||
ftgxCharData* glyphData = cacheGlyphData( text[i], pixelSize );
|
ftgxCharData* glyphData = cacheGlyphData(text[i], pixelSize);
|
||||||
|
|
||||||
if ( glyphData != NULL )
|
if (glyphData != NULL)
|
||||||
{
|
{
|
||||||
strMax = glyphData->renderOffsetMax > strMax ? glyphData->renderOffsetMax : strMax;
|
strMax = glyphData->renderOffsetMax > strMax ? glyphData->renderOffsetMax : strMax;
|
||||||
strMin = glyphData->renderOffsetMin < strMin ? glyphData->renderOffsetMin : strMin;
|
strMin = glyphData->renderOffsetMin < strMin ? glyphData->renderOffsetMin : strMin;
|
||||||
@ -508,10 +505,10 @@ void FreeTypeGX::getOffset( const wchar_t *text, int16_t pixelSize, uint16_t wid
|
|||||||
++i;
|
++i;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( ftPointSize != pixelSize )
|
if (ftPointSize != pixelSize)
|
||||||
{
|
{
|
||||||
ftPointSize = pixelSize;
|
ftPointSize = pixelSize;
|
||||||
FT_Set_Pixel_Sizes( ftFace, 0, ftPointSize );
|
FT_Set_Pixel_Sizes(ftFace, 0, ftPointSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
ftgxAlign[pixelSize].ascender = ftFace->size->metrics.ascender >> 6;
|
ftgxAlign[pixelSize].ascender = ftFace->size->metrics.ascender >> 6;
|
||||||
@ -532,34 +529,35 @@ void FreeTypeGX::getOffset( const wchar_t *text, int16_t pixelSize, uint16_t wid
|
|||||||
* @param screenY The screen Y coordinate at which to output the rendered texture.
|
* @param screenY The screen Y coordinate at which to output the rendered texture.
|
||||||
* @param color Color to apply to the texture.
|
* @param color Color to apply to the texture.
|
||||||
*/
|
*/
|
||||||
void FreeTypeGX::copyTextureToFramebuffer( GXTexObj *texObj, f32 texWidth, f32 texHeight, int16_t screenX, int16_t screenY, int16_t screenZ, GXColor color )
|
void FreeTypeGX::copyTextureToFramebuffer(GXTexObj *texObj, f32 texWidth, f32 texHeight, int16_t screenX,
|
||||||
|
int16_t screenY, int16_t screenZ, GXColor color)
|
||||||
{
|
{
|
||||||
GX_LoadTexObj( texObj, GX_TEXMAP0 );
|
GX_LoadTexObj(texObj, GX_TEXMAP0);
|
||||||
GX_InvalidateTexAll();
|
GX_InvalidateTexAll();
|
||||||
|
|
||||||
GX_SetTevOp ( GX_TEVSTAGE0, GX_MODULATE );
|
GX_SetTevOp(GX_TEVSTAGE0, GX_MODULATE);
|
||||||
GX_SetVtxDesc ( GX_VA_TEX0, GX_DIRECT );
|
GX_SetVtxDesc(GX_VA_TEX0, GX_DIRECT);
|
||||||
|
|
||||||
GX_Begin( GX_QUADS, this->vertexIndex, 4 );
|
GX_Begin(GX_QUADS, this->vertexIndex, 4);
|
||||||
GX_Position3s16( screenX, screenY, screenZ );
|
GX_Position3s16(screenX, screenY, screenZ);
|
||||||
GX_Color4u8( color.r, color.g, color.b, color.a );
|
GX_Color4u8(color.r, color.g, color.b, color.a);
|
||||||
GX_TexCoord2f32( 0.0f, 0.0f );
|
GX_TexCoord2f32(0.0f, 0.0f);
|
||||||
|
|
||||||
GX_Position3s16( texWidth + screenX, screenY, screenZ );
|
GX_Position3s16(texWidth + screenX, screenY, screenZ);
|
||||||
GX_Color4u8( color.r, color.g, color.b, color.a );
|
GX_Color4u8(color.r, color.g, color.b, color.a);
|
||||||
GX_TexCoord2f32( 1.0f, 0.0f );
|
GX_TexCoord2f32(1.0f, 0.0f);
|
||||||
|
|
||||||
GX_Position3s16( texWidth + screenX, texHeight + screenY, screenZ );
|
GX_Position3s16(texWidth + screenX, texHeight + screenY, screenZ);
|
||||||
GX_Color4u8( color.r, color.g, color.b, color.a );
|
GX_Color4u8(color.r, color.g, color.b, color.a);
|
||||||
GX_TexCoord2f32( 1.0f, 1.0f );
|
GX_TexCoord2f32(1.0f, 1.0f);
|
||||||
|
|
||||||
GX_Position3s16( screenX, texHeight + screenY, screenZ );
|
GX_Position3s16(screenX, texHeight + screenY, screenZ);
|
||||||
GX_Color4u8( color.r, color.g, color.b, color.a );
|
GX_Color4u8(color.r, color.g, color.b, color.a);
|
||||||
GX_TexCoord2f32( 0.0f, 1.0f );
|
GX_TexCoord2f32(0.0f, 1.0f);
|
||||||
GX_End();
|
GX_End();
|
||||||
|
|
||||||
GX_SetTevOp( GX_TEVSTAGE0, GX_PASSCLR );
|
GX_SetTevOp(GX_TEVSTAGE0, GX_PASSCLR);
|
||||||
GX_SetVtxDesc( GX_VA_TEX0, GX_NONE );
|
GX_SetVtxDesc(GX_VA_TEX0, GX_NONE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -573,25 +571,26 @@ void FreeTypeGX::copyTextureToFramebuffer( GXTexObj *texObj, f32 texWidth, f32 t
|
|||||||
* @param screenY The screen Y coordinate at which to output the quad.
|
* @param screenY The screen Y coordinate at which to output the quad.
|
||||||
* @param color Color to apply to the texture.
|
* @param color Color to apply to the texture.
|
||||||
*/
|
*/
|
||||||
void FreeTypeGX::copyFeatureToFramebuffer( f32 featureWidth, f32 featureHeight, int16_t screenX, int16_t screenY, int16_t screenZ, GXColor color )
|
void FreeTypeGX::copyFeatureToFramebuffer(f32 featureWidth, f32 featureHeight, int16_t screenX, int16_t screenY,
|
||||||
|
int16_t screenZ, GXColor color)
|
||||||
{
|
{
|
||||||
GX_SetTevOp ( GX_TEVSTAGE0, GX_PASSCLR );
|
GX_SetTevOp(GX_TEVSTAGE0, GX_PASSCLR);
|
||||||
GX_SetVtxDesc ( GX_VA_TEX0, GX_NONE );
|
GX_SetVtxDesc(GX_VA_TEX0, GX_NONE);
|
||||||
|
|
||||||
GX_Begin( GX_QUADS, this->vertexIndex, 4 );
|
GX_Begin(GX_QUADS, this->vertexIndex, 4);
|
||||||
GX_Position3s16( screenX, screenY, screenZ );
|
GX_Position3s16(screenX, screenY, screenZ);
|
||||||
GX_Color4u8( color.r, color.g, color.b, color.a );
|
GX_Color4u8(color.r, color.g, color.b, color.a);
|
||||||
|
|
||||||
GX_Position3s16( featureWidth + screenX, screenY, screenZ );
|
GX_Position3s16(featureWidth + screenX, screenY, screenZ);
|
||||||
GX_Color4u8( color.r, color.g, color.b, color.a );
|
GX_Color4u8(color.r, color.g, color.b, color.a);
|
||||||
|
|
||||||
GX_Position3s16( featureWidth + screenX, featureHeight + screenY, screenZ );
|
GX_Position3s16(featureWidth + screenX, featureHeight + screenY, screenZ);
|
||||||
GX_Color4u8( color.r, color.g, color.b, color.a );
|
GX_Color4u8(color.r, color.g, color.b, color.a);
|
||||||
|
|
||||||
GX_Position3s16( screenX, featureHeight + screenY, screenZ );
|
GX_Position3s16(screenX, featureHeight + screenY, screenZ);
|
||||||
GX_Color4u8( color.r, color.g, color.b, color.a );
|
GX_Color4u8(color.r, color.g, color.b, color.a);
|
||||||
GX_End();
|
GX_End();
|
||||||
|
|
||||||
GX_SetTevOp( GX_TEVSTAGE0, GX_PASSCLR );
|
GX_SetTevOp(GX_TEVSTAGE0, GX_PASSCLR);
|
||||||
GX_SetVtxDesc( GX_VA_TEX0, GX_NONE );
|
GX_SetVtxDesc(GX_VA_TEX0, GX_NONE);
|
||||||
}
|
}
|
||||||
|
@ -39,18 +39,18 @@
|
|||||||
*/
|
*/
|
||||||
typedef struct ftgxCharData_
|
typedef struct ftgxCharData_
|
||||||
{
|
{
|
||||||
int16_t renderOffsetX; /**< Texture X axis bearing offset. */
|
int16_t renderOffsetX; /**< Texture X axis bearing offset. */
|
||||||
uint16_t glyphAdvanceX; /**< Character glyph X coordinate advance in pixels. */
|
uint16_t glyphAdvanceX; /**< Character glyph X coordinate advance in pixels. */
|
||||||
uint32_t glyphIndex; /**< Charachter glyph index in the font face. */
|
uint32_t glyphIndex; /**< Charachter glyph index in the font face. */
|
||||||
|
|
||||||
uint16_t textureWidth; /**< Texture width in pixels/bytes. */
|
uint16_t textureWidth; /**< Texture width in pixels/bytes. */
|
||||||
uint16_t textureHeight; /**< Texture glyph height in pixels/bytes. */
|
uint16_t textureHeight; /**< Texture glyph height in pixels/bytes. */
|
||||||
|
|
||||||
int16_t renderOffsetY; /**< Texture Y axis bearing offset. */
|
int16_t renderOffsetY; /**< Texture Y axis bearing offset. */
|
||||||
int16_t renderOffsetMax; /**< Texture Y axis bearing maximum value. */
|
int16_t renderOffsetMax; /**< Texture Y axis bearing maximum value. */
|
||||||
int16_t renderOffsetMin; /**< Texture Y axis bearing minimum value. */
|
int16_t renderOffsetMin; /**< Texture Y axis bearing minimum value. */
|
||||||
|
|
||||||
uint8_t* glyphDataTexture; /**< Glyph texture bitmap data buffer. */
|
uint8_t* glyphDataTexture; /**< Glyph texture bitmap data buffer. */
|
||||||
} ftgxCharData;
|
} ftgxCharData;
|
||||||
|
|
||||||
/*! \struct ftgxDataOffset_
|
/*! \struct ftgxDataOffset_
|
||||||
@ -59,10 +59,10 @@ typedef struct ftgxCharData_
|
|||||||
*/
|
*/
|
||||||
typedef struct ftgxDataOffset_
|
typedef struct ftgxDataOffset_
|
||||||
{
|
{
|
||||||
int16_t ascender; /**< Maximum data offset. */
|
int16_t ascender; /**< Maximum data offset. */
|
||||||
int16_t descender; /**< Minimum data offset. */
|
int16_t descender; /**< Minimum data offset. */
|
||||||
int16_t max; /**< Maximum data offset. */
|
int16_t max; /**< Maximum data offset. */
|
||||||
int16_t min; /**< Minimum data offset. */
|
int16_t min; /**< Minimum data offset. */
|
||||||
} ftgxDataOffset;
|
} ftgxDataOffset;
|
||||||
|
|
||||||
typedef struct ftgxCharData_ ftgxCharData;
|
typedef struct ftgxCharData_ ftgxCharData;
|
||||||
@ -89,9 +89,10 @@ typedef struct ftgxDataOffset_ ftgxDataOffset;
|
|||||||
#define FTGX_STYLE_STRIKE 0x2000
|
#define FTGX_STYLE_STRIKE 0x2000
|
||||||
#define FTGX_STYLE_MASK 0xf000
|
#define FTGX_STYLE_MASK 0xf000
|
||||||
|
|
||||||
const GXColor ftgxWhite = ( GXColor ) {0xff, 0xff, 0xff, 0xff}; /**< Constant color value used only to sanitize Doxygen documentation. */
|
const GXColor ftgxWhite = ( GXColor )
|
||||||
|
{ 0xff, 0xff, 0xff, 0xff}; /**< Constant color value used only to sanitize Doxygen documentation. */
|
||||||
|
|
||||||
wchar_t* charToWideChar( const char* p );
|
wchar_t* charToWideChar(const char* p);
|
||||||
|
|
||||||
/*! \class FreeTypeGX
|
/*! \class FreeTypeGX
|
||||||
* \brief Wrapper class for the libFreeType library with GX rendering.
|
* \brief Wrapper class for the libFreeType library with GX rendering.
|
||||||
@ -105,40 +106,44 @@ wchar_t* charToWideChar( const char* p );
|
|||||||
class FreeTypeGX
|
class FreeTypeGX
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
FT_Library ftLibrary; /**< FreeType FT_Library instance. */
|
FT_Library ftLibrary; /**< FreeType FT_Library instance. */
|
||||||
FT_Face ftFace; /**< FreeType reusable FT_Face typographic object. */
|
FT_Face ftFace; /**< FreeType reusable FT_Face typographic object. */
|
||||||
int16_t ftPointSize; /**< Current set size of the rendered font. */
|
int16_t ftPointSize; /**< Current set size of the rendered font. */
|
||||||
bool ftKerningEnabled; /**< Flag indicating the availability of font kerning data. */
|
bool ftKerningEnabled; /**< Flag indicating the availability of font kerning data. */
|
||||||
uint8_t vertexIndex; /**< Vertex format descriptor index. */
|
uint8_t vertexIndex; /**< Vertex format descriptor index. */
|
||||||
std::map<int16_t, std::map<wchar_t, ftgxCharData> > fontData; /**< Map which holds the glyph data structures for the corresponding characters in one size. */
|
std::map<int16_t, std::map<wchar_t, ftgxCharData> > fontData; /**< Map which holds the glyph data structures for the corresponding characters in one size. */
|
||||||
std::map<int16_t, ftgxDataOffset> ftgxAlign; /**< Map which holds the ascender and decender for different sizes. */
|
std::map<int16_t, ftgxDataOffset> ftgxAlign; /**< Map which holds the ascender and decender for different sizes. */
|
||||||
|
|
||||||
int16_t getStyleOffsetWidth( uint16_t width, uint16_t format );
|
int16_t getStyleOffsetWidth(uint16_t width, uint16_t format);
|
||||||
int16_t getStyleOffsetHeight( int16_t format, uint16_t pixelSize );
|
int16_t getStyleOffsetHeight(int16_t format, uint16_t pixelSize);
|
||||||
|
|
||||||
void unloadFont();
|
void unloadFont();
|
||||||
ftgxCharData *cacheGlyphData( wchar_t charCode, int16_t pixelSize );
|
ftgxCharData *cacheGlyphData(wchar_t charCode, int16_t pixelSize);
|
||||||
uint16_t cacheGlyphDataComplete( int16_t pixelSize );
|
uint16_t cacheGlyphDataComplete(int16_t pixelSize);
|
||||||
void loadGlyphData( FT_Bitmap *bmp, ftgxCharData *charData );
|
void loadGlyphData(FT_Bitmap *bmp, ftgxCharData *charData);
|
||||||
|
|
||||||
void setDefaultMode();
|
void setDefaultMode();
|
||||||
|
|
||||||
void drawTextFeature( int16_t x, int16_t y, int16_t z, int16_t pixelSize, uint16_t width, ftgxDataOffset *offsetData, uint16_t format, GXColor color );
|
void drawTextFeature(int16_t x, int16_t y, int16_t z, int16_t pixelSize, uint16_t width,
|
||||||
void copyTextureToFramebuffer( GXTexObj *texObj, f32 texWidth, f32 texHeight, int16_t screenX, int16_t screenY, int16_t screenZ, GXColor color );
|
ftgxDataOffset *offsetData, uint16_t format, GXColor color);
|
||||||
void copyFeatureToFramebuffer( f32 featureWidth, f32 featureHeight, int16_t screenX, int16_t screenY, int16_t screenZ, GXColor color );
|
void copyTextureToFramebuffer(GXTexObj *texObj, f32 texWidth, f32 texHeight, int16_t screenX, int16_t screenY,
|
||||||
|
int16_t screenZ, GXColor color);
|
||||||
|
void copyFeatureToFramebuffer(f32 featureWidth, f32 featureHeight, int16_t screenX, int16_t screenY,
|
||||||
|
int16_t screenZ, GXColor color);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
FreeTypeGX( const uint8_t* fontBuffer, FT_Long bufferSize );
|
FreeTypeGX(const uint8_t* fontBuffer, FT_Long bufferSize);
|
||||||
~FreeTypeGX();
|
~FreeTypeGX();
|
||||||
|
|
||||||
void setVertexFormat( uint8_t vertexIndex );
|
void setVertexFormat(uint8_t vertexIndex);
|
||||||
|
|
||||||
uint16_t drawText( int16_t x, int16_t y, int16_t z, const wchar_t *text, int16_t pixelSize, GXColor color = ftgxWhite, uint16_t textStyling = FTGX_NULL, uint16_t textWidth = 0, uint16_t widthLimit = 0 );
|
uint16_t drawText(int16_t x, int16_t y, int16_t z, const wchar_t *text, int16_t pixelSize, GXColor color =
|
||||||
|
ftgxWhite, uint16_t textStyling = FTGX_NULL, uint16_t textWidth = 0, uint16_t widthLimit = 0);
|
||||||
|
|
||||||
uint16_t getWidth( const wchar_t *text, int16_t pixelSize );
|
uint16_t getWidth(const wchar_t *text, int16_t pixelSize);
|
||||||
uint16_t getCharWidth( const wchar_t wChar, int16_t pixelSize, const wchar_t prevChar = 0x0000 );
|
uint16_t getCharWidth(const wchar_t wChar, int16_t pixelSize, const wchar_t prevChar = 0x0000);
|
||||||
uint16_t getHeight( const wchar_t *text, int16_t pixelSize );
|
uint16_t getHeight(const wchar_t *text, int16_t pixelSize);
|
||||||
void getOffset( const wchar_t *text, int16_t pixelSize, uint16_t widthLimit = 0 );
|
void getOffset(const wchar_t *text, int16_t pixelSize, uint16_t widthLimit = 0);
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* FREETYPEGX_H_ */
|
#endif /* FREETYPEGX_H_ */
|
||||||
|
@ -39,16 +39,15 @@
|
|||||||
#include "ZipFile.h"
|
#include "ZipFile.h"
|
||||||
#include "language/gettext.h"
|
#include "language/gettext.h"
|
||||||
|
|
||||||
ZipFile::ZipFile( const char *filepath )
|
ZipFile::ZipFile(const char *filepath)
|
||||||
{
|
{
|
||||||
File = unzOpen( filepath );
|
File = unzOpen(filepath);
|
||||||
if ( File )
|
if (File) this->LoadList();
|
||||||
this->LoadList();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ZipFile::~ZipFile()
|
ZipFile::~ZipFile()
|
||||||
{
|
{
|
||||||
unzClose( File );
|
unzClose(File);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ZipFile::LoadList()
|
bool ZipFile::LoadList()
|
||||||
@ -56,83 +55,76 @@ bool ZipFile::LoadList()
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ZipFile::ExtractAll( const char *dest )
|
bool ZipFile::ExtractAll(const char *dest)
|
||||||
{
|
{
|
||||||
if ( !File )
|
if (!File) return false;
|
||||||
return false;
|
|
||||||
|
|
||||||
bool Stop = false;
|
bool Stop = false;
|
||||||
|
|
||||||
u32 blocksize = 1024 * 50;
|
u32 blocksize = 1024 * 50;
|
||||||
u8 *buffer = new u8[blocksize];
|
u8 *buffer = new u8[blocksize];
|
||||||
|
|
||||||
if ( !buffer )
|
if (!buffer) return false;
|
||||||
return false;
|
|
||||||
|
|
||||||
char writepath[MAXPATHLEN];
|
char writepath[MAXPATHLEN];
|
||||||
char filename[MAXPATHLEN];
|
char filename[MAXPATHLEN];
|
||||||
memset( filename, 0, sizeof( filename ) );
|
memset(filename, 0, sizeof(filename));
|
||||||
|
|
||||||
int ret = unzGoToFirstFile( File );
|
int ret = unzGoToFirstFile(File);
|
||||||
if ( ret != UNZ_OK )
|
if (ret != UNZ_OK) Stop = true;
|
||||||
Stop = true;
|
|
||||||
|
|
||||||
while ( !Stop )
|
while (!Stop)
|
||||||
{
|
{
|
||||||
if ( unzGetCurrentFileInfo( File, &cur_file_info, filename, sizeof( filename ), NULL, NULL, NULL, NULL ) != UNZ_OK )
|
if (unzGetCurrentFileInfo(File, &cur_file_info, filename, sizeof(filename), NULL, NULL, NULL, NULL) != UNZ_OK) Stop
|
||||||
Stop = true;
|
= true;
|
||||||
|
|
||||||
if ( !Stop && filename[strlen( filename )-1] != '/' )
|
if (!Stop && filename[strlen(filename) - 1] != '/')
|
||||||
{
|
{
|
||||||
u32 uncompressed_size = cur_file_info.uncompressed_size;
|
u32 uncompressed_size = cur_file_info.uncompressed_size;
|
||||||
|
|
||||||
u32 done = 0;
|
u32 done = 0;
|
||||||
char *pointer = NULL;
|
char *pointer = NULL;
|
||||||
|
|
||||||
ret = unzOpenCurrentFile( File );
|
ret = unzOpenCurrentFile(File);
|
||||||
|
|
||||||
snprintf( writepath, sizeof( writepath ), "%s/%s", dest, filename );
|
snprintf(writepath, sizeof(writepath), "%s/%s", dest, filename);
|
||||||
|
|
||||||
pointer = strrchr( writepath, '/' );
|
pointer = strrchr(writepath, '/');
|
||||||
int position = pointer - writepath + 2;
|
int position = pointer - writepath + 2;
|
||||||
|
|
||||||
char temppath[strlen( writepath )];
|
char temppath[strlen(writepath)];
|
||||||
snprintf( temppath, position, "%s", writepath );
|
snprintf(temppath, position, "%s", writepath);
|
||||||
|
|
||||||
subfoldercreate( temppath );
|
subfoldercreate(temppath);
|
||||||
|
|
||||||
if ( ret == UNZ_OK )
|
if (ret == UNZ_OK)
|
||||||
{
|
{
|
||||||
FILE *pfile = fopen( writepath, "wb" );
|
FILE *pfile = fopen(writepath, "wb");
|
||||||
|
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
ShowProgress( tr( "Extracting files..." ), 0, pointer + 1, done, uncompressed_size );
|
ShowProgress(tr( "Extracting files..." ), 0, pointer + 1, done, uncompressed_size);
|
||||||
|
|
||||||
if ( uncompressed_size - done < blocksize )
|
if (uncompressed_size - done < blocksize) blocksize = uncompressed_size - done;
|
||||||
blocksize = uncompressed_size - done;
|
|
||||||
|
|
||||||
ret = unzReadCurrentFile( File, buffer, blocksize );
|
ret = unzReadCurrentFile(File, buffer, blocksize);
|
||||||
|
|
||||||
if ( ret == 0 )
|
if (ret == 0) break;
|
||||||
break;
|
|
||||||
|
|
||||||
fwrite( buffer, 1, blocksize, pfile );
|
fwrite(buffer, 1, blocksize, pfile);
|
||||||
|
|
||||||
done += ret;
|
done += ret;
|
||||||
|
|
||||||
}
|
} while (done < uncompressed_size);
|
||||||
while ( done < uncompressed_size );
|
|
||||||
|
|
||||||
fclose( pfile );
|
fclose(pfile);
|
||||||
unzCloseCurrentFile( File );
|
unzCloseCurrentFile(File);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ( unzGoToNextFile( File ) != UNZ_OK )
|
if (unzGoToNextFile(File) != UNZ_OK) Stop = true;
|
||||||
Stop = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
delete [] buffer;
|
delete[] buffer;
|
||||||
buffer = NULL;
|
buffer = NULL;
|
||||||
|
|
||||||
ProgressStop();
|
ProgressStop();
|
||||||
|
@ -32,22 +32,22 @@
|
|||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
u64 offset; // ZipFile offset
|
u64 offset; // ZipFile offset
|
||||||
u64 length; // uncompressed file length in 64 bytes for sizes higher than 4GB
|
u64 length; // uncompressed file length in 64 bytes for sizes higher than 4GB
|
||||||
bool isdir; // 0 - file, 1 - directory
|
bool isdir; // 0 - file, 1 - directory
|
||||||
char filename[256]; // full filename
|
char filename[256]; // full filename
|
||||||
} FileStructure;
|
} FileStructure;
|
||||||
|
|
||||||
class ZipFile
|
class ZipFile
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
//!Constructor
|
//!Constructor
|
||||||
ZipFile( const char *filepath );
|
ZipFile(const char *filepath);
|
||||||
//!Destructor
|
//!Destructor
|
||||||
~ZipFile();
|
~ZipFile();
|
||||||
//!Extract all files from a zip file to a directory
|
//!Extract all files from a zip file to a directory
|
||||||
//!\param dest Destination path to where to extract
|
//!\param dest Destination path to where to extract
|
||||||
bool ExtractAll( const char *dest );
|
bool ExtractAll(const char *dest);
|
||||||
protected:
|
protected:
|
||||||
bool LoadList();
|
bool LoadList();
|
||||||
unzFile File;
|
unzFile File;
|
||||||
|
@ -17,9 +17,9 @@
|
|||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
void InitAudio()
|
void InitAudio()
|
||||||
{
|
{
|
||||||
AUDIO_Init( NULL );
|
AUDIO_Init(NULL);
|
||||||
ASND_Init();
|
ASND_Init();
|
||||||
ASND_Pause( 0 );
|
ASND_Pause(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@ -30,6 +30,6 @@ void InitAudio()
|
|||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
void ShutdownAudio()
|
void ShutdownAudio()
|
||||||
{
|
{
|
||||||
ASND_Pause( 1 );
|
ASND_Pause(1);
|
||||||
ASND_End();
|
ASND_End();
|
||||||
}
|
}
|
||||||
|
@ -86,7 +86,6 @@
|
|||||||
|
|
||||||
#include "MD5.h"
|
#include "MD5.h"
|
||||||
|
|
||||||
|
|
||||||
/* -------------------------------------------------------------------------- **
|
/* -------------------------------------------------------------------------- **
|
||||||
* Static Constants:
|
* Static Constants:
|
||||||
*
|
*
|
||||||
@ -110,46 +109,34 @@
|
|||||||
* array. They're divided up into four groups of 16.
|
* array. They're divided up into four groups of 16.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static const uint8_t K[3][16] =
|
static const uint8_t K[3][16] = {
|
||||||
{
|
/* Round 1: skipped (since it is simply sequential). */
|
||||||
/* Round 1: skipped (since it is simply sequential). */
|
{ 1, 6, 11, 0, 5, 10, 15, 4, 9, 14, 3, 8, 13, 2, 7, 12 }, /* R2 */
|
||||||
{ 1, 6, 11, 0, 5, 10, 15, 4, 9, 14, 3, 8, 13, 2, 7, 12 }, /* R2 */
|
{ 5, 8, 11, 14, 1, 4, 7, 10, 13, 0, 3, 6, 9, 12, 15, 2 }, /* R3 */
|
||||||
{ 5, 8, 11, 14, 1, 4, 7, 10, 13, 0, 3, 6, 9, 12, 15, 2 }, /* R3 */
|
{ 0, 7, 14, 5, 12, 3, 10, 1, 8, 15, 6, 13, 4, 11, 2, 9 } /* R4 */
|
||||||
{ 0, 7, 14, 5, 12, 3, 10, 1, 8, 15, 6, 13, 4, 11, 2, 9 } /* R4 */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static const uint8_t S[4][4] =
|
static const uint8_t S[4][4] = { { 7, 12, 17, 22 }, /* Round 1 */
|
||||||
{
|
{ 5, 9, 14, 20 }, /* Round 2 */
|
||||||
{ 7, 12, 17, 22 }, /* Round 1 */
|
{ 4, 11, 16, 23 }, /* Round 3 */
|
||||||
{ 5, 9, 14, 20 }, /* Round 2 */
|
{ 6, 10, 15, 21 } /* Round 4 */
|
||||||
{ 4, 11, 16, 23 }, /* Round 3 */
|
|
||||||
{ 6, 10, 15, 21 } /* Round 4 */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const uint32_t T[4][16] = { { 0xd76aa478, 0xe8c7b756, 0x242070db, 0xc1bdceee, /* Round 1 */
|
||||||
|
0xf57c0faf, 0x4787c62a, 0xa8304613, 0xfd469501, 0x698098d8, 0x8b44f7af, 0xffff5bb1, 0x895cd7be, 0x6b901122, 0xfd987193,
|
||||||
|
0xa679438e, 0x49b40821 },
|
||||||
|
|
||||||
static const uint32_t T[4][16] =
|
{ 0xf61e2562, 0xc040b340, 0x265e5a51, 0xe9b6c7aa, /* Round 2 */
|
||||||
{
|
0xd62f105d, 0x02441453, 0xd8a1e681, 0xe7d3fbc8, 0x21e1cde6, 0xc33707d6, 0xf4d50d87, 0x455a14ed, 0xa9e3e905, 0xfcefa3f8,
|
||||||
{ 0xd76aa478, 0xe8c7b756, 0x242070db, 0xc1bdceee, /* Round 1 */
|
0x676f02d9, 0x8d2a4c8a },
|
||||||
0xf57c0faf, 0x4787c62a, 0xa8304613, 0xfd469501,
|
|
||||||
0x698098d8, 0x8b44f7af, 0xffff5bb1, 0x895cd7be,
|
|
||||||
0x6b901122, 0xfd987193, 0xa679438e, 0x49b40821 },
|
|
||||||
|
|
||||||
{ 0xf61e2562, 0xc040b340, 0x265e5a51, 0xe9b6c7aa, /* Round 2 */
|
{ 0xfffa3942, 0x8771f681, 0x6d9d6122, 0xfde5380c, /* Round 3 */
|
||||||
0xd62f105d, 0x02441453, 0xd8a1e681, 0xe7d3fbc8,
|
0xa4beea44, 0x4bdecfa9, 0xf6bb4b60, 0xbebfbc70, 0x289b7ec6, 0xeaa127fa, 0xd4ef3085, 0x04881d05, 0xd9d4d039, 0xe6db99e5,
|
||||||
0x21e1cde6, 0xc33707d6, 0xf4d50d87, 0x455a14ed,
|
0x1fa27cf8, 0xc4ac5665 },
|
||||||
0xa9e3e905, 0xfcefa3f8, 0x676f02d9, 0x8d2a4c8a },
|
|
||||||
|
|
||||||
{ 0xfffa3942, 0x8771f681, 0x6d9d6122, 0xfde5380c, /* Round 3 */
|
|
||||||
0xa4beea44, 0x4bdecfa9, 0xf6bb4b60, 0xbebfbc70,
|
|
||||||
0x289b7ec6, 0xeaa127fa, 0xd4ef3085, 0x04881d05,
|
|
||||||
0xd9d4d039, 0xe6db99e5, 0x1fa27cf8, 0xc4ac5665 },
|
|
||||||
|
|
||||||
{ 0xf4292244, 0x432aff97, 0xab9423a7, 0xfc93a039, /* Round 4 */
|
|
||||||
0x655b59c3, 0x8f0ccc92, 0xffeff47d, 0x85845dd1,
|
|
||||||
0x6fa87e4f, 0xfe2ce6e0, 0xa3014314, 0x4e0811a1,
|
|
||||||
0xf7537e82, 0xbd3af235, 0x2ad7d2bb, 0xeb86d391 },
|
|
||||||
};
|
|
||||||
|
|
||||||
|
{ 0xf4292244, 0x432aff97, 0xab9423a7, 0xfc93a039, /* Round 4 */
|
||||||
|
0x655b59c3, 0x8f0ccc92, 0xffeff47d, 0x85845dd1, 0x6fa87e4f, 0xfe2ce6e0, 0xa3014314, 0x4e0811a1, 0xf7537e82, 0xbd3af235,
|
||||||
|
0x2ad7d2bb, 0xeb86d391 }, };
|
||||||
|
|
||||||
/* -------------------------------------------------------------------------- **
|
/* -------------------------------------------------------------------------- **
|
||||||
* Macros:
|
* Macros:
|
||||||
@ -172,12 +159,11 @@ static const uint32_t T[4][16] =
|
|||||||
|
|
||||||
#define STR2HEX(x) ((x >= 0x30) && (x <= 0x39)) ? x - 0x30 : toupper((int)x)-0x37
|
#define STR2HEX(x) ((x >= 0x30) && (x <= 0x39)) ? x - 0x30 : toupper((int)x)-0x37
|
||||||
|
|
||||||
|
|
||||||
/* -------------------------------------------------------------------------- **
|
/* -------------------------------------------------------------------------- **
|
||||||
* Static Functions:
|
* Static Functions:
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static void Permute( uint32_t ABCD[4], const unsigned char block[64] )
|
static void Permute(uint32_t ABCD[4], const unsigned char block[64])
|
||||||
/* ------------------------------------------------------------------------ **
|
/* ------------------------------------------------------------------------ **
|
||||||
* Permute the ABCD "registers" using the 64-byte <block> as a driver.
|
* Permute the ABCD "registers" using the 64-byte <block> as a driver.
|
||||||
*
|
*
|
||||||
@ -211,28 +197,28 @@ static void Permute( uint32_t ABCD[4], const unsigned char block[64] )
|
|||||||
* ------------------------------------------------------------------------ **
|
* ------------------------------------------------------------------------ **
|
||||||
*/
|
*/
|
||||||
{
|
{
|
||||||
int round;
|
int round;
|
||||||
int i, j;
|
int i, j;
|
||||||
uint8_t s;
|
uint8_t s;
|
||||||
uint32_t a, b, c, d;
|
uint32_t a, b, c, d;
|
||||||
uint32_t KeepABCD[4];
|
uint32_t KeepABCD[4];
|
||||||
uint32_t X[16];
|
uint32_t X[16];
|
||||||
|
|
||||||
/* Store the current ABCD values for later re-use.
|
/* Store the current ABCD values for later re-use.
|
||||||
*/
|
*/
|
||||||
for ( i = 0; i < 4; i++ )
|
for (i = 0; i < 4; i++)
|
||||||
KeepABCD[i] = ABCD[i];
|
KeepABCD[i] = ABCD[i];
|
||||||
|
|
||||||
/* Convert the input block into an array of unsigned longs, taking care
|
/* Convert the input block into an array of unsigned longs, taking care
|
||||||
* to read the block in Little Endian order (the algorithm assumes this).
|
* to read the block in Little Endian order (the algorithm assumes this).
|
||||||
* The uint32_t values are then handled in host order.
|
* The uint32_t values are then handled in host order.
|
||||||
*/
|
*/
|
||||||
for ( i = 0, j = 0; i < 16; i++ )
|
for (i = 0, j = 0; i < 16; i++)
|
||||||
{
|
{
|
||||||
X[i] = ( uint32_t )block[j++];
|
X[i] = (uint32_t) block[j++];
|
||||||
X[i] |= ( ( uint32_t )block[j++] << 8 );
|
X[i] |= ((uint32_t) block[j++] << 8);
|
||||||
X[i] |= ( ( uint32_t )block[j++] << 16 );
|
X[i] |= ((uint32_t) block[j++] << 16);
|
||||||
X[i] |= ( ( uint32_t )block[j++] << 24 );
|
X[i] |= ((uint32_t) block[j++] << 24);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This loop performs the four rounds of permutations.
|
/* This loop performs the four rounds of permutations.
|
||||||
@ -249,21 +235,21 @@ static void Permute( uint32_t ABCD[4], const unsigned char block[64] )
|
|||||||
* (My implementation appears to be a poor compromise between speed, size,
|
* (My implementation appears to be a poor compromise between speed, size,
|
||||||
* and clarity. Ugh. [crh])
|
* and clarity. Ugh. [crh])
|
||||||
*/
|
*/
|
||||||
for ( round = 0; round < 4; round++ )
|
for (round = 0; round < 4; round++)
|
||||||
{
|
{
|
||||||
for ( i = 0; i < 16; i++ )
|
for (i = 0; i < 16; i++)
|
||||||
{
|
{
|
||||||
j = ( 4 - ( i % 4 ) ) & 0x3; /* <j> handles the rotation of ABCD. */
|
j = (4 - (i % 4)) & 0x3; /* <j> handles the rotation of ABCD. */
|
||||||
s = S[round][i%4]; /* <s> is the bit shift for this iteration. */
|
s = S[round][i % 4]; /* <s> is the bit shift for this iteration. */
|
||||||
|
|
||||||
b = ABCD[( j+1 ) & 0x3]; /* Copy the b,c,d values per ABCD rotation. */
|
b = ABCD[(j + 1) & 0x3]; /* Copy the b,c,d values per ABCD rotation. */
|
||||||
c = ABCD[( j+2 ) & 0x3]; /* This isn't really necessary, it just looks */
|
c = ABCD[(j + 2) & 0x3]; /* This isn't really necessary, it just looks */
|
||||||
d = ABCD[( j+3 ) & 0x3]; /* clean & will hopefully be optimized away. */
|
d = ABCD[(j + 3) & 0x3]; /* clean & will hopefully be optimized away. */
|
||||||
|
|
||||||
/* The actual perumation function.
|
/* The actual perumation function.
|
||||||
* This is broken out to minimize the code within the switch().
|
* This is broken out to minimize the code within the switch().
|
||||||
*/
|
*/
|
||||||
switch ( round )
|
switch (round)
|
||||||
{
|
{
|
||||||
case 0:
|
case 0:
|
||||||
/* round 1 */
|
/* round 1 */
|
||||||
@ -271,36 +257,35 @@ static void Permute( uint32_t ABCD[4], const unsigned char block[64] )
|
|||||||
break;
|
break;
|
||||||
case 1:
|
case 1:
|
||||||
/* round 2 */
|
/* round 2 */
|
||||||
a = md5G( b, c, d ) + X[ K[0][i] ];
|
a = md5G( b, c, d ) + X[K[0][i]];
|
||||||
break;
|
break;
|
||||||
case 2:
|
case 2:
|
||||||
/* round 3 */
|
/* round 3 */
|
||||||
a = md5H( b, c, d ) + X[ K[1][i] ];
|
a = md5H( b, c, d ) + X[K[1][i]];
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
/* round 4 */
|
/* round 4 */
|
||||||
a = md5I( b, c, d ) + X[ K[2][i] ];
|
a = md5I( b, c, d ) + X[K[2][i]];
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
a = 0xFFFFFFFF & ( ABCD[j] + a + T[round][i] );
|
a = 0xFFFFFFFF & (ABCD[j] + a + T[round][i]);
|
||||||
ABCD[j] = b + ( 0xFFFFFFFF & ( ( a << s ) | ( a >> ( 32 - s ) ) ) );
|
ABCD[j] = b + (0xFFFFFFFF & ((a << s) | (a >> (32 - s))));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Use the stored original A, B, C, D values to perform
|
/* Use the stored original A, B, C, D values to perform
|
||||||
* one last convolution.
|
* one last convolution.
|
||||||
*/
|
*/
|
||||||
for ( i = 0; i < 4; i++ )
|
for (i = 0; i < 4; i++)
|
||||||
ABCD[i] = 0xFFFFFFFF & ( ABCD[i] + KeepABCD[i] );
|
ABCD[i] = 0xFFFFFFFF & (ABCD[i] + KeepABCD[i]);
|
||||||
|
|
||||||
} /* Permute */
|
} /* Permute */
|
||||||
|
|
||||||
|
|
||||||
/* -------------------------------------------------------------------------- **
|
/* -------------------------------------------------------------------------- **
|
||||||
* Functions:
|
* Functions:
|
||||||
*/
|
*/
|
||||||
|
|
||||||
auth_md5Ctx *auth_md5InitCtx( auth_md5Ctx *ctx )
|
auth_md5Ctx *auth_md5InitCtx(auth_md5Ctx *ctx)
|
||||||
/* ------------------------------------------------------------------------ **
|
/* ------------------------------------------------------------------------ **
|
||||||
* Initialize an MD5 context.
|
* Initialize an MD5 context.
|
||||||
*
|
*
|
||||||
@ -329,13 +314,13 @@ auth_md5Ctx *auth_md5InitCtx( auth_md5Ctx *ctx )
|
|||||||
* ------------------------------------------------------------------------ **
|
* ------------------------------------------------------------------------ **
|
||||||
*/
|
*/
|
||||||
{
|
{
|
||||||
ctx->len = 0;
|
ctx->len = 0;
|
||||||
ctx->b_used = 0;
|
ctx->b_used = 0;
|
||||||
|
|
||||||
ctx->ABCD[0] = 0x67452301; /* The array ABCD[] contains the four 4-byte */
|
ctx->ABCD[0] = 0x67452301; /* The array ABCD[] contains the four 4-byte */
|
||||||
ctx->ABCD[1] = 0xefcdab89; /* "registers" that are manipulated to */
|
ctx->ABCD[1] = 0xefcdab89; /* "registers" that are manipulated to */
|
||||||
ctx->ABCD[2] = 0x98badcfe; /* produce the MD5 digest. The input acts */
|
ctx->ABCD[2] = 0x98badcfe; /* produce the MD5 digest. The input acts */
|
||||||
ctx->ABCD[3] = 0x10325476; /* upon the registers, not the other way */
|
ctx->ABCD[3] = 0x10325476; /* upon the registers, not the other way */
|
||||||
/* 'round. The initial values are those */
|
/* 'round. The initial values are those */
|
||||||
/* given in RFC 1321 (pg. 4). Note, however, that RFC 1321 */
|
/* given in RFC 1321 (pg. 4). Note, however, that RFC 1321 */
|
||||||
/* provides these values as bytes, not as longwords, and the */
|
/* provides these values as bytes, not as longwords, and the */
|
||||||
@ -344,13 +329,10 @@ auth_md5Ctx *auth_md5InitCtx( auth_md5Ctx *ctx )
|
|||||||
/* confusing as all getout (to me, anyway). The values given */
|
/* confusing as all getout (to me, anyway). The values given */
|
||||||
/* here are provided as 32-bit values in C language format, */
|
/* here are provided as 32-bit values in C language format, */
|
||||||
/* so they are endian-agnostic. */
|
/* so they are endian-agnostic. */
|
||||||
return( ctx );
|
return (ctx);
|
||||||
} /* auth_md5InitCtx */
|
} /* auth_md5InitCtx */
|
||||||
|
|
||||||
|
auth_md5Ctx *auth_md5SumCtx(auth_md5Ctx *ctx, const unsigned char *src, const int len)
|
||||||
auth_md5Ctx *auth_md5SumCtx( auth_md5Ctx *ctx,
|
|
||||||
const unsigned char *src,
|
|
||||||
const int len )
|
|
||||||
/* ------------------------------------------------------------------------ **
|
/* ------------------------------------------------------------------------ **
|
||||||
* Build an MD5 Message Digest within the given context.
|
* Build an MD5 Message Digest within the given context.
|
||||||
*
|
*
|
||||||
@ -371,29 +353,28 @@ auth_md5Ctx *auth_md5SumCtx( auth_md5Ctx *ctx,
|
|||||||
|
|
||||||
/* Add the new block's length to the total length.
|
/* Add the new block's length to the total length.
|
||||||
*/
|
*/
|
||||||
ctx->len += ( uint32_t )len;
|
ctx->len += (uint32_t) len;
|
||||||
|
|
||||||
/* Copy the new block's data into the context block.
|
/* Copy the new block's data into the context block.
|
||||||
* Call the Permute() function whenever the context block is full.
|
* Call the Permute() function whenever the context block is full.
|
||||||
*/
|
*/
|
||||||
for ( i = 0; i < len; i++ )
|
for (i = 0; i < len; i++)
|
||||||
{
|
{
|
||||||
ctx->block[ ctx->b_used ] = src[i];
|
ctx->block[ctx->b_used] = src[i];
|
||||||
( ctx->b_used )++;
|
(ctx->b_used)++;
|
||||||
if ( 64 == ctx->b_used )
|
if (64 == ctx->b_used)
|
||||||
{
|
{
|
||||||
Permute( ctx->ABCD, ctx->block );
|
Permute(ctx->ABCD, ctx->block);
|
||||||
ctx->b_used = 0;
|
ctx->b_used = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the updated context.
|
/* Return the updated context.
|
||||||
*/
|
*/
|
||||||
return( ctx );
|
return (ctx);
|
||||||
} /* auth_md5SumCtx */
|
} /* auth_md5SumCtx */
|
||||||
|
|
||||||
|
auth_md5Ctx *auth_md5CloseCtx(auth_md5Ctx *ctx, unsigned char *dst)
|
||||||
auth_md5Ctx *auth_md5CloseCtx( auth_md5Ctx *ctx, unsigned char *dst )
|
|
||||||
/* ------------------------------------------------------------------------ **
|
/* ------------------------------------------------------------------------ **
|
||||||
* Close an MD5 Message Digest context and generate the final MD5 sum.
|
* Close an MD5 Message Digest context and generate the final MD5 sum.
|
||||||
*
|
*
|
||||||
@ -413,7 +394,7 @@ auth_md5Ctx *auth_md5CloseCtx( auth_md5Ctx *ctx, unsigned char *dst )
|
|||||||
* ------------------------------------------------------------------------ **
|
* ------------------------------------------------------------------------ **
|
||||||
*/
|
*/
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
uint32_t l;
|
uint32_t l;
|
||||||
|
|
||||||
/* Add the required 0x80 padding initiator byte.
|
/* Add the required 0x80 padding initiator byte.
|
||||||
@ -422,20 +403,20 @@ auth_md5Ctx *auth_md5CloseCtx( auth_md5Ctx *ctx, unsigned char *dst )
|
|||||||
* free byte in the context block.
|
* free byte in the context block.
|
||||||
*/
|
*/
|
||||||
ctx->block[ctx->b_used] = 0x80;
|
ctx->block[ctx->b_used] = 0x80;
|
||||||
( ctx->b_used )++;
|
(ctx->b_used)++;
|
||||||
|
|
||||||
/* Zero out any remaining free bytes in the context block.
|
/* Zero out any remaining free bytes in the context block.
|
||||||
*/
|
*/
|
||||||
for ( i = ctx->b_used; i < 64; i++ )
|
for (i = ctx->b_used; i < 64; i++)
|
||||||
ctx->block[i] = 0;
|
ctx->block[i] = 0;
|
||||||
|
|
||||||
/* We need 8 bytes to store the length field.
|
/* We need 8 bytes to store the length field.
|
||||||
* If we don't have 8, call Permute() and reset the context block.
|
* If we don't have 8, call Permute() and reset the context block.
|
||||||
*/
|
*/
|
||||||
if ( 56 < ctx->b_used )
|
if (56 < ctx->b_used)
|
||||||
{
|
{
|
||||||
Permute( ctx->ABCD, ctx->block );
|
Permute(ctx->ABCD, ctx->block);
|
||||||
for ( i = 0; i < 64; i++ )
|
for (i = 0; i < 64; i++)
|
||||||
ctx->block[i] = 0;
|
ctx->block[i] = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -445,29 +426,28 @@ auth_md5Ctx *auth_md5CloseCtx( auth_md5Ctx *ctx, unsigned char *dst )
|
|||||||
* any MAXINT numeric overflow issues.
|
* any MAXINT numeric overflow issues.
|
||||||
*/
|
*/
|
||||||
l = ctx->len << 3;
|
l = ctx->len << 3;
|
||||||
for ( i = 0; i < 4; i++ )
|
for (i = 0; i < 4; i++)
|
||||||
ctx->block[56+i] |= GetLongByte( l, i );
|
ctx->block[56 + i] |= GetLongByte( l, i );
|
||||||
ctx->block[60] = ( ( GetLongByte( ctx->len, 3 ) & 0xE0 ) >> 5 ); /* See Above! */
|
ctx->block[60] = ((GetLongByte( ctx->len, 3 ) & 0xE0) >> 5); /* See Above! */
|
||||||
Permute( ctx->ABCD, ctx->block );
|
Permute(ctx->ABCD, ctx->block);
|
||||||
|
|
||||||
/* Now copy the result into the output buffer and we're done.
|
/* Now copy the result into the output buffer and we're done.
|
||||||
*/
|
*/
|
||||||
for ( i = 0; i < 4; i++ )
|
for (i = 0; i < 4; i++)
|
||||||
{
|
{
|
||||||
dst[ 0+i] = GetLongByte( ctx->ABCD[0], i );
|
dst[0 + i] = GetLongByte( ctx->ABCD[0], i );
|
||||||
dst[ 4+i] = GetLongByte( ctx->ABCD[1], i );
|
dst[4 + i] = GetLongByte( ctx->ABCD[1], i );
|
||||||
dst[ 8+i] = GetLongByte( ctx->ABCD[2], i );
|
dst[8 + i] = GetLongByte( ctx->ABCD[2], i );
|
||||||
dst[12+i] = GetLongByte( ctx->ABCD[3], i );
|
dst[12 + i] = GetLongByte( ctx->ABCD[3], i );
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the context.
|
/* Return the context.
|
||||||
* This is done for compatibility with the other auth_md5*Ctx() functions.
|
* This is done for compatibility with the other auth_md5*Ctx() functions.
|
||||||
*/
|
*/
|
||||||
return( ctx );
|
return (ctx);
|
||||||
} /* auth_md5CloseCtx */
|
} /* auth_md5CloseCtx */
|
||||||
|
|
||||||
|
unsigned char * MD5(unsigned char *dst, const unsigned char *src, const int len)
|
||||||
unsigned char * MD5( unsigned char *dst, const unsigned char *src, const int len )
|
|
||||||
/* ------------------------------------------------------------------------ **
|
/* ------------------------------------------------------------------------ **
|
||||||
* Compute an MD5 message digest.
|
* Compute an MD5 message digest.
|
||||||
*
|
*
|
||||||
@ -503,16 +483,14 @@ unsigned char * MD5( unsigned char *dst, const unsigned char *src, const int len
|
|||||||
{
|
{
|
||||||
auth_md5Ctx ctx[1];
|
auth_md5Ctx ctx[1];
|
||||||
|
|
||||||
( void )auth_md5InitCtx( ctx ); /* Open a context. */
|
(void) auth_md5InitCtx(ctx); /* Open a context. */
|
||||||
( void )auth_md5SumCtx( ctx, src, len ); /* Pass only one block. */
|
(void) auth_md5SumCtx(ctx, src, len); /* Pass only one block. */
|
||||||
( void )auth_md5CloseCtx( ctx, dst ); /* Close the context. */
|
(void) auth_md5CloseCtx(ctx, dst); /* Close the context. */
|
||||||
|
|
||||||
return( dst ); /* Makes life easy. */
|
return (dst); /* Makes life easy. */
|
||||||
} /* auth_md5Sum */
|
} /* auth_md5Sum */
|
||||||
|
|
||||||
|
unsigned char * MD5fromFile(unsigned char *dst, const char *src)
|
||||||
|
|
||||||
unsigned char * MD5fromFile( unsigned char *dst, const char *src )
|
|
||||||
/* ------------------------------------------------------------------------ **
|
/* ------------------------------------------------------------------------ **
|
||||||
* Compute an MD5 message digest.
|
* Compute an MD5 message digest.
|
||||||
*
|
*
|
||||||
@ -550,58 +528,55 @@ unsigned char * MD5fromFile( unsigned char *dst, const char *src )
|
|||||||
unsigned int blksize = 0;
|
unsigned int blksize = 0;
|
||||||
unsigned int read = 0;
|
unsigned int read = 0;
|
||||||
|
|
||||||
file = fopen( src, "rb" );
|
file = fopen(src, "rb");
|
||||||
|
|
||||||
if ( file == NULL )
|
if (file == NULL)
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
( void )auth_md5InitCtx( ctx ); /* Open a context. */
|
(void) auth_md5InitCtx(ctx); /* Open a context. */
|
||||||
|
|
||||||
fseek ( file , 0 , SEEK_END );
|
fseek(file, 0, SEEK_END);
|
||||||
unsigned long long filesize = ftell( file );
|
unsigned long long filesize = ftell(file);
|
||||||
rewind ( file );
|
rewind(file);
|
||||||
|
|
||||||
if ( filesize < 1048576 ) //1MB cache for files bigger than 1 MB
|
if (filesize < 1048576) //1MB cache for files bigger than 1 MB
|
||||||
blksize = filesize;
|
blksize = filesize;
|
||||||
else
|
else blksize = 1048576;
|
||||||
blksize = 1048576;
|
|
||||||
|
|
||||||
unsigned char * buffer = malloc( blksize );
|
unsigned char * buffer = malloc(blksize);
|
||||||
|
|
||||||
if ( buffer == NULL )
|
if (buffer == NULL)
|
||||||
{
|
{
|
||||||
//no memory
|
//no memory
|
||||||
fclose( file );
|
fclose(file);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
read = fread( buffer, 1, blksize, file );
|
read = fread(buffer, 1, blksize, file);
|
||||||
( void )auth_md5SumCtx( ctx, buffer, read ); /* Pass only one block. */
|
(void) auth_md5SumCtx(ctx, buffer, read); /* Pass only one block. */
|
||||||
|
|
||||||
}
|
} while (read > 0);
|
||||||
while ( read > 0 );
|
|
||||||
|
|
||||||
fclose( file );
|
fclose(file);
|
||||||
free( buffer );
|
free(buffer);
|
||||||
|
|
||||||
( void )auth_md5CloseCtx( ctx, dst ); /* Close the context. */
|
(void) auth_md5CloseCtx(ctx, dst); /* Close the context. */
|
||||||
|
|
||||||
return( dst ); /* Makes life easy. */
|
return (dst); /* Makes life easy. */
|
||||||
} /* auth_md5Sum */
|
} /* auth_md5Sum */
|
||||||
|
|
||||||
|
const char * MD5ToString(const unsigned char * hash, char * dst)
|
||||||
const char * MD5ToString( const unsigned char * hash, char * dst )
|
|
||||||
{
|
{
|
||||||
char hexchar[3];
|
char hexchar[3];
|
||||||
short i = 0, n = 0;
|
short i = 0, n = 0;
|
||||||
|
|
||||||
for ( i = 0; i < 16; i++ )
|
for (i = 0; i < 16; i++)
|
||||||
{
|
{
|
||||||
sprintf( hexchar, "%02X", hash[i] );
|
sprintf(hexchar, "%02X", hash[i]);
|
||||||
|
|
||||||
dst[n++] = hexchar[0];
|
dst[n++] = hexchar[0];
|
||||||
dst[n++] = hexchar[1];
|
dst[n++] = hexchar[1];
|
||||||
@ -612,12 +587,12 @@ const char * MD5ToString( const unsigned char * hash, char * dst )
|
|||||||
return dst;
|
return dst;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned char * StringToMD5( const char * hash, unsigned char * dst )
|
unsigned char * StringToMD5(const char * hash, unsigned char * dst)
|
||||||
{
|
{
|
||||||
char hexchar[2];
|
char hexchar[2];
|
||||||
short i = 0, n = 0;
|
short i = 0, n = 0;
|
||||||
|
|
||||||
for ( i = 0; i < 16; i++ )
|
for (i = 0; i < 16; i++)
|
||||||
{
|
{
|
||||||
hexchar[0] = hash[n++];
|
hexchar[0] = hash[n++];
|
||||||
hexchar[1] = hash[n++];
|
hexchar[1] = hash[n++];
|
||||||
@ -630,5 +605,4 @@ unsigned char * StringToMD5( const char * hash, unsigned char * dst )
|
|||||||
return dst;
|
return dst;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* ========================================================================== */
|
/* ========================================================================== */
|
||||||
|
@ -5,242 +5,237 @@
|
|||||||
extern "C"
|
extern "C"
|
||||||
{
|
{
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* ========================================================================== **
|
/* ========================================================================== **
|
||||||
*
|
*
|
||||||
* MD5.h
|
* MD5.h
|
||||||
*
|
*
|
||||||
* Copyright:
|
* Copyright:
|
||||||
* Copyright (C) 2003-2005 by Christopher R. Hertel
|
* Copyright (C) 2003-2005 by Christopher R. Hertel
|
||||||
*
|
*
|
||||||
* Email: crh@ubiqx.mn.org
|
* Email: crh@ubiqx.mn.org
|
||||||
*
|
*
|
||||||
* $Id: MD5.h,v 0.6 2005/06/08 18:35:59 crh Exp $
|
* $Id: MD5.h,v 0.6 2005/06/08 18:35:59 crh Exp $
|
||||||
*
|
*
|
||||||
* Modifications and additions by dimok
|
* Modifications and additions by dimok
|
||||||
*
|
*
|
||||||
* -------------------------------------------------------------------------- **
|
* -------------------------------------------------------------------------- **
|
||||||
*
|
*
|
||||||
* Description:
|
* Description:
|
||||||
* Implements the MD5 hash algorithm, as described in RFC 1321.
|
* Implements the MD5 hash algorithm, as described in RFC 1321.
|
||||||
*
|
*
|
||||||
* -------------------------------------------------------------------------- **
|
* -------------------------------------------------------------------------- **
|
||||||
*
|
*
|
||||||
* License:
|
* License:
|
||||||
*
|
*
|
||||||
* This library is free software; you can redistribute it and/or
|
* This library is free software; you can redistribute it and/or
|
||||||
* modify it under the terms of the GNU Lesser General Public
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
* License as published by the Free Software Foundation; either
|
* License as published by the Free Software Foundation; either
|
||||||
* version 2.1 of the License, or (at your option) any later version.
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
*
|
*
|
||||||
* This library is distributed in the hope that it will be useful,
|
* This library is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
* Lesser General Public License for more details.
|
* Lesser General Public License for more details.
|
||||||
*
|
*
|
||||||
* You should have received a copy of the GNU Lesser General Public
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
* License along with this library; if not, write to the Free Software
|
* License along with this library; if not, write to the Free Software
|
||||||
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
*
|
*
|
||||||
* -------------------------------------------------------------------------- **
|
* -------------------------------------------------------------------------- **
|
||||||
*
|
*
|
||||||
* Notes:
|
* Notes:
|
||||||
*
|
*
|
||||||
* None of this will make any sense unless you're studying RFC 1321 as you
|
* None of this will make any sense unless you're studying RFC 1321 as you
|
||||||
* read the code.
|
* read the code.
|
||||||
*
|
*
|
||||||
* MD5 is described in RFC 1321.
|
* MD5 is described in RFC 1321.
|
||||||
* The MD*4* algorithm is described in RFC 1320 (that's 1321 - 1).
|
* The MD*4* algorithm is described in RFC 1320 (that's 1321 - 1).
|
||||||
* MD5 is very similar to MD4, but not quite similar enough to justify
|
* MD5 is very similar to MD4, but not quite similar enough to justify
|
||||||
* putting the two into a single module. Besides, I wanted to add a few
|
* putting the two into a single module. Besides, I wanted to add a few
|
||||||
* extra functions to this one to expand its usability.
|
* extra functions to this one to expand its usability.
|
||||||
*
|
*
|
||||||
* There are three primary motivations for this particular implementation.
|
* There are three primary motivations for this particular implementation.
|
||||||
* 1) Programmer's pride. I wanted to be able to say I'd done it, and I
|
* 1) Programmer's pride. I wanted to be able to say I'd done it, and I
|
||||||
* wanted to learn from the experience.
|
* wanted to learn from the experience.
|
||||||
* 2) Portability. I wanted an implementation that I knew to be portable
|
* 2) Portability. I wanted an implementation that I knew to be portable
|
||||||
* to a reasonable number of platforms. In particular, the algorithm is
|
* to a reasonable number of platforms. In particular, the algorithm is
|
||||||
* designed with little-endian platforms in mind, but I wanted an
|
* designed with little-endian platforms in mind, but I wanted an
|
||||||
* endian-agnostic implementation.
|
* endian-agnostic implementation.
|
||||||
* 3) Compactness. While not an overriding goal, I thought it worth-while
|
* 3) Compactness. While not an overriding goal, I thought it worth-while
|
||||||
* to see if I could reduce the overall size of the result. This is in
|
* to see if I could reduce the overall size of the result. This is in
|
||||||
* keeping with my hopes that this library will be suitable for use in
|
* keeping with my hopes that this library will be suitable for use in
|
||||||
* some embedded environments.
|
* some embedded environments.
|
||||||
* Beyond that, cleanliness and clarity are always worth pursuing.
|
* Beyond that, cleanliness and clarity are always worth pursuing.
|
||||||
*
|
*
|
||||||
* As mentioned above, the code really only makes sense if you are familiar
|
* As mentioned above, the code really only makes sense if you are familiar
|
||||||
* with the MD5 algorithm or are using RFC 1321 as a guide. This code is
|
* with the MD5 algorithm or are using RFC 1321 as a guide. This code is
|
||||||
* quirky, however, so you'll want to be reading carefully.
|
* quirky, however, so you'll want to be reading carefully.
|
||||||
*
|
*
|
||||||
* Yeah...most of the comments are cut-and-paste from my MD4 implementation.
|
* Yeah...most of the comments are cut-and-paste from my MD4 implementation.
|
||||||
*
|
*
|
||||||
* -------------------------------------------------------------------------- **
|
* -------------------------------------------------------------------------- **
|
||||||
*
|
*
|
||||||
* References:
|
* References:
|
||||||
* IETF RFC 1321: The MD5 Message-Digest Algorithm
|
* IETF RFC 1321: The MD5 Message-Digest Algorithm
|
||||||
* Ron Rivest. IETF, April, 1992
|
* Ron Rivest. IETF, April, 1992
|
||||||
*
|
*
|
||||||
* ========================================================================== **
|
* ========================================================================== **
|
||||||
*/
|
*/
|
||||||
/* -------------------------------------------------------------------------- **
|
/* -------------------------------------------------------------------------- **
|
||||||
* Typedefs:
|
* Typedefs:
|
||||||
*/
|
*/
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
unsigned int len;
|
unsigned int len;
|
||||||
unsigned int ABCD[4];
|
unsigned int ABCD[4];
|
||||||
int b_used;
|
int b_used;
|
||||||
unsigned char block[64];
|
unsigned char block[64];
|
||||||
} auth_md5Ctx;
|
} auth_md5Ctx;
|
||||||
|
|
||||||
|
/* -------------------------------------------------------------------------- **
|
||||||
/* -------------------------------------------------------------------------- **
|
* Functions:
|
||||||
* Functions:
|
*/
|
||||||
*/
|
|
||||||
|
auth_md5Ctx *auth_md5InitCtx(auth_md5Ctx *ctx);
|
||||||
auth_md5Ctx *auth_md5InitCtx( auth_md5Ctx *ctx );
|
/* ------------------------------------------------------------------------ **
|
||||||
/* ------------------------------------------------------------------------ **
|
* Initialize an MD5 context.
|
||||||
* Initialize an MD5 context.
|
*
|
||||||
*
|
* Input: ctx - A pointer to the MD5 context structure to be initialized.
|
||||||
* Input: ctx - A pointer to the MD5 context structure to be initialized.
|
* Contexts are typically created thusly:
|
||||||
* Contexts are typically created thusly:
|
* ctx = (auth_md5Ctx *)malloc( sizeof(auth_md5Ctx) );
|
||||||
* ctx = (auth_md5Ctx *)malloc( sizeof(auth_md5Ctx) );
|
*
|
||||||
*
|
* Output: A pointer to the initialized context (same as <ctx>).
|
||||||
* Output: A pointer to the initialized context (same as <ctx>).
|
*
|
||||||
*
|
* Notes: The purpose of the context is to make it possible to generate
|
||||||
* Notes: The purpose of the context is to make it possible to generate
|
* an MD5 Message Digest in stages, rather than having to pass a
|
||||||
* an MD5 Message Digest in stages, rather than having to pass a
|
* single large block to a single MD5 function. The context
|
||||||
* single large block to a single MD5 function. The context
|
* structure keeps track of various bits of state information.
|
||||||
* structure keeps track of various bits of state information.
|
*
|
||||||
*
|
* Once the context is initialized, the blocks of message data
|
||||||
* Once the context is initialized, the blocks of message data
|
* are passed to the <auth_md5SumCtx()> function. Once the
|
||||||
* are passed to the <auth_md5SumCtx()> function. Once the
|
* final bit of data has been handed to <auth_md5SumCtx()> the
|
||||||
* final bit of data has been handed to <auth_md5SumCtx()> the
|
* context can be closed out by calling <auth_md5CloseCtx()>,
|
||||||
* context can be closed out by calling <auth_md5CloseCtx()>,
|
* which also calculates the final MD5 result.
|
||||||
* which also calculates the final MD5 result.
|
*
|
||||||
*
|
* Don't forget to free an allocated context structure when
|
||||||
* Don't forget to free an allocated context structure when
|
* you've finished using it.
|
||||||
* you've finished using it.
|
*
|
||||||
*
|
* See Also: <auth_md5SumCtx()>, <auth_md5CloseCtx()>
|
||||||
* See Also: <auth_md5SumCtx()>, <auth_md5CloseCtx()>
|
*
|
||||||
*
|
* ------------------------------------------------------------------------ **
|
||||||
* ------------------------------------------------------------------------ **
|
*/
|
||||||
*/
|
|
||||||
|
auth_md5Ctx *auth_md5SumCtx(auth_md5Ctx *ctx, const unsigned char *src, const int len);
|
||||||
|
/* ------------------------------------------------------------------------ **
|
||||||
auth_md5Ctx *auth_md5SumCtx( auth_md5Ctx *ctx,
|
* Build an MD5 Message Digest within the given context.
|
||||||
const unsigned char *src,
|
*
|
||||||
const int len );
|
* Input: ctx - Pointer to the context in which the MD5 sum is being
|
||||||
/* ------------------------------------------------------------------------ **
|
* built.
|
||||||
* Build an MD5 Message Digest within the given context.
|
* src - A chunk of source data. This will be used to drive
|
||||||
*
|
* the MD5 algorithm.
|
||||||
* Input: ctx - Pointer to the context in which the MD5 sum is being
|
* len - The number of bytes in <src>.
|
||||||
* built.
|
*
|
||||||
* src - A chunk of source data. This will be used to drive
|
* Output: A pointer to the updated context (same as <ctx>).
|
||||||
* the MD5 algorithm.
|
*
|
||||||
* len - The number of bytes in <src>.
|
* See Also: <auth_md5InitCtx()>, <auth_md5CloseCtx()>, <auth_md5Sum()>
|
||||||
*
|
*
|
||||||
* Output: A pointer to the updated context (same as <ctx>).
|
* ------------------------------------------------------------------------ **
|
||||||
*
|
*/
|
||||||
* See Also: <auth_md5InitCtx()>, <auth_md5CloseCtx()>, <auth_md5Sum()>
|
|
||||||
*
|
auth_md5Ctx *auth_md5CloseCtx(auth_md5Ctx *ctx, unsigned char *dst);
|
||||||
* ------------------------------------------------------------------------ **
|
/* ------------------------------------------------------------------------ **
|
||||||
*/
|
* Close an MD5 Message Digest context and generate the final MD5 sum.
|
||||||
|
*
|
||||||
|
* Input: ctx - Pointer to the context in which the MD5 sum is being
|
||||||
auth_md5Ctx *auth_md5CloseCtx( auth_md5Ctx *ctx, unsigned char *dst );
|
* built.
|
||||||
/* ------------------------------------------------------------------------ **
|
* dst - A pointer to at least 16 bytes of memory, which will
|
||||||
* Close an MD5 Message Digest context and generate the final MD5 sum.
|
* receive the finished MD5 sum.
|
||||||
*
|
*
|
||||||
* Input: ctx - Pointer to the context in which the MD5 sum is being
|
* Output: A pointer to the closed context (same as <ctx>).
|
||||||
* built.
|
* You might use this to free a malloc'd context structure. :)
|
||||||
* dst - A pointer to at least 16 bytes of memory, which will
|
*
|
||||||
* receive the finished MD5 sum.
|
* Notes: The context (<ctx>) is returned in an undefined state.
|
||||||
*
|
* It must be re-initialized before re-use.
|
||||||
* Output: A pointer to the closed context (same as <ctx>).
|
*
|
||||||
* You might use this to free a malloc'd context structure. :)
|
* See Also: <auth_md5InitCtx()>, <auth_md5SumCtx()>
|
||||||
*
|
*
|
||||||
* Notes: The context (<ctx>) is returned in an undefined state.
|
* ------------------------------------------------------------------------ **
|
||||||
* It must be re-initialized before re-use.
|
*/
|
||||||
*
|
|
||||||
* See Also: <auth_md5InitCtx()>, <auth_md5SumCtx()>
|
unsigned char * MD5(unsigned char * hash, const unsigned char *src, const int len);
|
||||||
*
|
/* ------------------------------------------------------------------------ **
|
||||||
* ------------------------------------------------------------------------ **
|
* Compute an MD5 message digest.
|
||||||
*/
|
*
|
||||||
|
* Input: dst - Destination buffer into which the result will be written.
|
||||||
|
* Must be 16 bytes, minimum.
|
||||||
unsigned char * MD5( unsigned char * hash, const unsigned char *src, const int len );
|
* src - Source data block to be MD5'd.
|
||||||
/* ------------------------------------------------------------------------ **
|
* len - The length, in bytes, of the source block.
|
||||||
* Compute an MD5 message digest.
|
* (Note that the length is given in bytes, not bits.)
|
||||||
*
|
*
|
||||||
* Input: dst - Destination buffer into which the result will be written.
|
* Output: A pointer to <dst>, which will contain the calculated 16-byte
|
||||||
* Must be 16 bytes, minimum.
|
* MD5 message digest.
|
||||||
* src - Source data block to be MD5'd.
|
*
|
||||||
* len - The length, in bytes, of the source block.
|
* Notes: This function is a shortcut. It takes a single input block.
|
||||||
* (Note that the length is given in bytes, not bits.)
|
* For more drawn-out operations, see <auth_md5InitCtx()>.
|
||||||
*
|
*
|
||||||
* Output: A pointer to <dst>, which will contain the calculated 16-byte
|
* This function is interface-compatible with the
|
||||||
* MD5 message digest.
|
* <auth_md4Sum()> function in the MD4 module.
|
||||||
*
|
*
|
||||||
* Notes: This function is a shortcut. It takes a single input block.
|
* The MD5 algorithm is designed to work on data with an
|
||||||
* For more drawn-out operations, see <auth_md5InitCtx()>.
|
* arbitrary *bit* length. Most implementations, this one
|
||||||
*
|
* included, handle the input data in byte-sized chunks.
|
||||||
* This function is interface-compatible with the
|
*
|
||||||
* <auth_md4Sum()> function in the MD4 module.
|
* The MD5 algorithm does much of its work using four-byte
|
||||||
*
|
* words, and so can be tuned for speed based on the endian-ness
|
||||||
* The MD5 algorithm is designed to work on data with an
|
* of the host. This implementation is intended to be
|
||||||
* arbitrary *bit* length. Most implementations, this one
|
* endian-neutral, which may make it a teeny bit slower than
|
||||||
* included, handle the input data in byte-sized chunks.
|
* others. ...maybe.
|
||||||
*
|
*
|
||||||
* The MD5 algorithm does much of its work using four-byte
|
* See Also: <auth_md5InitCtx()>
|
||||||
* words, and so can be tuned for speed based on the endian-ness
|
*
|
||||||
* of the host. This implementation is intended to be
|
* ------------------------------------------------------------------------ **
|
||||||
* endian-neutral, which may make it a teeny bit slower than
|
*/
|
||||||
* others. ...maybe.
|
|
||||||
*
|
unsigned char * MD5fromFile(unsigned char *dst, const char *src);
|
||||||
* See Also: <auth_md5InitCtx()>
|
/* ------------------------------------------------------------------------ **
|
||||||
*
|
* Compute an MD5 message digest.
|
||||||
* ------------------------------------------------------------------------ **
|
*
|
||||||
*/
|
* Input: dst - Destination buffer into which the result will be written.
|
||||||
|
* Must be 16 bytes, minimum.
|
||||||
unsigned char * MD5fromFile( unsigned char *dst, const char *src );
|
* src - filepath to the file to be MD5'd.
|
||||||
/* ------------------------------------------------------------------------ **
|
*
|
||||||
* Compute an MD5 message digest.
|
* Output: A pointer to <dst>, which will contain the calculated 16-byte
|
||||||
*
|
* MD5 message digest.
|
||||||
* Input: dst - Destination buffer into which the result will be written.
|
*
|
||||||
* Must be 16 bytes, minimum.
|
* Notes: This function is a shortcut. It takes a single input block.
|
||||||
* src - filepath to the file to be MD5'd.
|
* For more drawn-out operations, see <auth_md5InitCtx()>.
|
||||||
*
|
*
|
||||||
* Output: A pointer to <dst>, which will contain the calculated 16-byte
|
* This function is interface-compatible with the
|
||||||
* MD5 message digest.
|
* <auth_md4Sum()> function in the MD4 module.
|
||||||
*
|
*
|
||||||
* Notes: This function is a shortcut. It takes a single input block.
|
* The MD5 algorithm is designed to work on data with an
|
||||||
* For more drawn-out operations, see <auth_md5InitCtx()>.
|
* arbitrary *bit* length. Most implementations, this one
|
||||||
*
|
* included, handle the input data in byte-sized chunks.
|
||||||
* This function is interface-compatible with the
|
*
|
||||||
* <auth_md4Sum()> function in the MD4 module.
|
* The MD5 algorithm does much of its work using four-byte
|
||||||
*
|
* words, and so can be tuned for speed based on the endian-ness
|
||||||
* The MD5 algorithm is designed to work on data with an
|
* of the host. This implementation is intended to be
|
||||||
* arbitrary *bit* length. Most implementations, this one
|
* endian-neutral, which may make it a teeny bit slower than
|
||||||
* included, handle the input data in byte-sized chunks.
|
* others. ...maybe.
|
||||||
*
|
*
|
||||||
* The MD5 algorithm does much of its work using four-byte
|
* See Also: <auth_md5InitCtx()>
|
||||||
* words, and so can be tuned for speed based on the endian-ness
|
*
|
||||||
* of the host. This implementation is intended to be
|
* ------------------------------------------------------------------------ **
|
||||||
* endian-neutral, which may make it a teeny bit slower than
|
*/
|
||||||
* others. ...maybe.
|
|
||||||
*
|
const char * MD5ToString(const unsigned char *hash, char *dst);
|
||||||
* See Also: <auth_md5InitCtx()>
|
unsigned char * StringToMD5(const char * hash, unsigned char * dst);
|
||||||
*
|
|
||||||
* ------------------------------------------------------------------------ **
|
/* ========================================================================== */
|
||||||
*/
|
|
||||||
|
#ifdef __cplusplus
|
||||||
const char * MD5ToString( const unsigned char *hash, char *dst );
|
}
|
||||||
unsigned char * StringToMD5( const char * hash, unsigned char * dst );
|
#endif
|
||||||
|
#endif /* AUTH_MD5_H */
|
||||||
/* ========================================================================== */
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#endif /* AUTH_MD5_H */
|
|
||||||
|
@ -22,106 +22,102 @@
|
|||||||
#include "patches/fst.h"
|
#include "patches/fst.h"
|
||||||
#include "usbloader/fstfile.h"
|
#include "usbloader/fstfile.h"
|
||||||
|
|
||||||
s32 dump_banner( const u8* discid, const char * dest )
|
s32 dump_banner(const u8* discid, const char * dest)
|
||||||
{
|
{
|
||||||
// Mount the disc
|
// Mount the disc
|
||||||
//Disc_SetWBFS(1, (u8*)discid);
|
//Disc_SetWBFS(1, (u8*)discid);
|
||||||
Disc_SetUSB( discid );
|
Disc_SetUSB(discid);
|
||||||
|
|
||||||
Disc_Open();
|
Disc_Open();
|
||||||
|
|
||||||
u64 offset;
|
u64 offset;
|
||||||
s32 ret;
|
s32 ret;
|
||||||
|
|
||||||
ret = __Disc_FindPartition( &offset );
|
ret = __Disc_FindPartition(&offset);
|
||||||
if ( ret < 0 )
|
if (ret < 0) return ret;
|
||||||
return ret;
|
|
||||||
|
|
||||||
ret = WDVD_OpenPartition( offset );
|
ret = WDVD_OpenPartition(offset);
|
||||||
|
|
||||||
if ( ret < 0 )
|
if (ret < 0)
|
||||||
{
|
{
|
||||||
//printf("ERROR: OpenPartition(0x%llx) %d\n", offset, ret);
|
//printf("ERROR: OpenPartition(0x%llx) %d\n", offset, ret);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read where to find the fst.bin
|
// Read where to find the fst.bin
|
||||||
u32 *buffer = memalign( 32, 0x20 );
|
u32 *buffer = memalign(32, 0x20);
|
||||||
|
|
||||||
if ( buffer == NULL )
|
if (buffer == NULL)
|
||||||
{
|
{
|
||||||
//Out of memory
|
//Out of memory
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = WDVD_Read( buffer, 0x20, 0x420 );
|
ret = WDVD_Read(buffer, 0x20, 0x420);
|
||||||
if ( ret < 0 )
|
if (ret < 0) return ret;
|
||||||
return ret;
|
|
||||||
|
|
||||||
// Read fst.bin
|
// Read fst.bin
|
||||||
void *fstbuffer = memalign( 32, buffer[2] * 4 );
|
void *fstbuffer = memalign(32, buffer[2] * 4);
|
||||||
FST_ENTRY *fst = ( FST_ENTRY * )fstbuffer;
|
FST_ENTRY *fst = (FST_ENTRY *) fstbuffer;
|
||||||
|
|
||||||
if ( fst == NULL )
|
if (fst == NULL)
|
||||||
{
|
{
|
||||||
//Out of memory
|
//Out of memory
|
||||||
free( buffer );
|
free(buffer);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = WDVD_Read( fstbuffer, buffer[2] * 4, buffer[1] * 4 );
|
ret = WDVD_Read(fstbuffer, buffer[2] * 4, buffer[1] * 4);
|
||||||
if ( ret < 0 )
|
if (ret < 0) return ret;
|
||||||
return ret;
|
|
||||||
|
|
||||||
free( buffer );
|
free(buffer);
|
||||||
|
|
||||||
// Search the fst.bin
|
// Search the fst.bin
|
||||||
u32 count = fst[0].filelen;
|
u32 count = fst[0].filelen;
|
||||||
int i;
|
int i;
|
||||||
u32 index = 0;
|
u32 index = 0;
|
||||||
|
|
||||||
for ( i = 1; i < count; i++ )
|
for (i = 1; i < count; i++)
|
||||||
{
|
{
|
||||||
if ( strstr( fstfiles( fst, i ), "opening.bnr" ) != NULL )
|
if (strstr(fstfiles(fst, i), "opening.bnr") != NULL)
|
||||||
{
|
{
|
||||||
index = i;
|
index = i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( index == 0 )
|
if (index == 0)
|
||||||
{
|
{
|
||||||
//opening.bnr not found
|
//opening.bnr not found
|
||||||
free( fstbuffer );
|
free(fstbuffer);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Load the .bnr
|
// Load the .bnr
|
||||||
u8 *banner = memalign( 32, fst[index].filelen );
|
u8 *banner = memalign(32, fst[index].filelen);
|
||||||
|
|
||||||
if ( banner == NULL )
|
if (banner == NULL)
|
||||||
{
|
{
|
||||||
//Out of memory
|
//Out of memory
|
||||||
free( fstbuffer );
|
free(fstbuffer);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = WDVD_Read( ( void * )banner, fst[index].filelen, fst[index].fileoffset * 4 );
|
ret = WDVD_Read((void *) banner, fst[index].filelen, fst[index].fileoffset * 4);
|
||||||
if ( ret < 0 )
|
if (ret < 0) return ret;
|
||||||
return ret;
|
|
||||||
|
|
||||||
WDVD_Reset();
|
WDVD_Reset();
|
||||||
WDVD_ClosePartition();
|
WDVD_ClosePartition();
|
||||||
//fatInitDefault();
|
//fatInitDefault();
|
||||||
//SDCard_Init();
|
//SDCard_Init();
|
||||||
WDVD_SetUSBMode( NULL, 0 );
|
WDVD_SetUSBMode(NULL, 0);
|
||||||
FILE *fp = fopen( dest, "wb" );
|
FILE *fp = fopen(dest, "wb");
|
||||||
if ( fp )
|
if (fp)
|
||||||
{
|
{
|
||||||
fwrite( banner, 1, fst[index].filelen, fp );
|
fwrite(banner, 1, fst[index].filelen, fp);
|
||||||
fclose( fp );
|
fclose(fp);
|
||||||
}
|
}
|
||||||
free( fstbuffer );
|
free(fstbuffer);
|
||||||
free( banner );
|
free(banner);
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -13,7 +13,7 @@ extern "C"
|
|||||||
{
|
{
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
s32 dump_banner( const u8 *discid, const char * dest );
|
s32 dump_banner(const u8 *discid, const char * dest);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
|
@ -6,118 +6,117 @@
|
|||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
#include "gui_banner.h"
|
#include "gui_banner.h"
|
||||||
|
|
||||||
GuiBanner::GuiBanner( const char *tplfilepath )
|
GuiBanner::GuiBanner(const char *tplfilepath)
|
||||||
{
|
{
|
||||||
memory = NULL;
|
memory = NULL;
|
||||||
tplfilesize = 0;
|
tplfilesize = 0;
|
||||||
width = 0;
|
width = 0;
|
||||||
height = 0;
|
height = 0;
|
||||||
|
|
||||||
FILE *tplfp = fopen( tplfilepath, "rb" );
|
FILE *tplfp = fopen(tplfilepath, "rb");
|
||||||
|
|
||||||
if ( tplfp != NULL )
|
if (tplfp != NULL)
|
||||||
{
|
{
|
||||||
|
|
||||||
unsigned short heighttemp = 0;
|
unsigned short heighttemp = 0;
|
||||||
unsigned short widthtemp = 0;
|
unsigned short widthtemp = 0;
|
||||||
|
|
||||||
fseek( tplfp , 0x14, SEEK_SET );
|
fseek(tplfp, 0x14, SEEK_SET);
|
||||||
fread( ( void* )&heighttemp, 1, 2, tplfp );
|
fread((void*) &heighttemp, 1, 2, tplfp);
|
||||||
fread( ( void* )&widthtemp, 1, 2, tplfp );
|
fread((void*) &widthtemp, 1, 2, tplfp);
|
||||||
fseek ( tplfp , 0 , SEEK_END );
|
fseek(tplfp, 0, SEEK_END);
|
||||||
tplfilesize = ftell ( tplfp );
|
tplfilesize = ftell(tplfp);
|
||||||
rewind ( tplfp );
|
rewind(tplfp);
|
||||||
memory = memalign( 32, tplfilesize );
|
memory = memalign(32, tplfilesize);
|
||||||
if ( !memory )
|
if (!memory)
|
||||||
{
|
{
|
||||||
fclose( tplfp );
|
fclose(tplfp);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
fread( memory, 1, tplfilesize, tplfp );
|
fread(memory, 1, tplfilesize, tplfp);
|
||||||
fclose( tplfp );
|
fclose(tplfp);
|
||||||
|
|
||||||
TPLFile tplfile;
|
TPLFile tplfile;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ret = TPL_OpenTPLFromMemory( &tplfile, memory, tplfilesize );
|
ret = TPL_OpenTPLFromMemory(&tplfile, memory, tplfilesize);
|
||||||
if ( ret < 0 )
|
if (ret < 0)
|
||||||
{
|
{
|
||||||
free( memory );
|
free(memory);
|
||||||
memory = NULL;
|
memory = NULL;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
ret = TPL_GetTexture( &tplfile, 0, &texObj );
|
ret = TPL_GetTexture(&tplfile, 0, &texObj);
|
||||||
if ( ret < 0 )
|
if (ret < 0)
|
||||||
{
|
{
|
||||||
free( memory );
|
free(memory);
|
||||||
memory = NULL;
|
memory = NULL;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
TPL_CloseTPLFile( &tplfile );
|
TPL_CloseTPLFile(&tplfile);
|
||||||
|
|
||||||
width = widthtemp;
|
width = widthtemp;
|
||||||
height = heighttemp;
|
height = heighttemp;
|
||||||
widescreen = 0;
|
widescreen = 0;
|
||||||
filecheck = true;
|
filecheck = true;
|
||||||
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
filecheck = false;
|
filecheck = false;
|
||||||
fclose( tplfp );
|
fclose(tplfp);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
GuiBanner::GuiBanner( void *mem, u32 len, int w, int h )
|
GuiBanner::GuiBanner(void *mem, u32 len, int w, int h)
|
||||||
{
|
{
|
||||||
if ( !mem || !len )
|
if (!mem || !len) return;
|
||||||
return;
|
memory = mem;
|
||||||
memory = mem;
|
tplfilesize = len;
|
||||||
tplfilesize = len;
|
width = w;
|
||||||
width = w;
|
height = h;
|
||||||
height = h;
|
|
||||||
|
TPLFile tplfile;
|
||||||
TPLFile tplfile;
|
|
||||||
|
int ret;
|
||||||
int ret;
|
|
||||||
|
ret = TPL_OpenTPLFromMemory(&tplfile, memory, tplfilesize);
|
||||||
ret = TPL_OpenTPLFromMemory( &tplfile, memory, tplfilesize );
|
if (ret < 0)
|
||||||
if ( ret < 0 )
|
{
|
||||||
{
|
free(memory);
|
||||||
free( memory );
|
memory = NULL;
|
||||||
memory = NULL;
|
return;
|
||||||
return;
|
}
|
||||||
}
|
ret = TPL_GetTexture(&tplfile, 0, &texObj);
|
||||||
ret = TPL_GetTexture( &tplfile, 0, &texObj );
|
if (ret < 0)
|
||||||
if ( ret < 0 )
|
{
|
||||||
{
|
free(memory);
|
||||||
free( memory );
|
memory = NULL;
|
||||||
memory = NULL;
|
return;
|
||||||
return;
|
}
|
||||||
}
|
TPL_CloseTPLFile(&tplfile);
|
||||||
TPL_CloseTPLFile( &tplfile );
|
|
||||||
|
filecheck = true;
|
||||||
filecheck = true;
|
}
|
||||||
}
|
|
||||||
|
GuiBanner::~GuiBanner()
|
||||||
GuiBanner::~GuiBanner()
|
{
|
||||||
{
|
if (memory != NULL)
|
||||||
if ( memory != NULL )
|
{
|
||||||
{
|
free(memory);
|
||||||
free( memory );
|
memory = NULL;
|
||||||
memory = NULL;
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
void GuiBanner::Draw()
|
||||||
void GuiBanner::Draw()
|
{
|
||||||
{
|
LOCK( this );
|
||||||
LOCK( this );
|
if (!filecheck || !this->IsVisible()) return;
|
||||||
if ( !filecheck || !this->IsVisible() )
|
|
||||||
return;
|
float currScale = this->GetScale();
|
||||||
|
|
||||||
float currScale = this->GetScale();
|
Menu_DrawTPLImg(this->GetLeft(), this->GetTop(), 0, width, height, &texObj, imageangle, widescreen ? currScale
|
||||||
|
* 0.80 : currScale, currScale, this->GetAlpha(), xx1, yy1, xx2, yy2, xx3, yy3, xx4, yy4);
|
||||||
Menu_DrawTPLImg( this->GetLeft(), this->GetTop(), 0, width, height, &texObj, imageangle, widescreen ? currScale*0.80 : currScale, currScale, this->GetAlpha(), xx1, yy1, xx2, yy2, xx3, yy3, xx4, yy4 );
|
|
||||||
|
this->UpdateEffects();
|
||||||
this->UpdateEffects();
|
}
|
||||||
}
|
|
||||||
|
@ -10,18 +10,18 @@
|
|||||||
|
|
||||||
#include "libwiigui/gui.h"
|
#include "libwiigui/gui.h"
|
||||||
|
|
||||||
class GuiBanner : public GuiImage
|
class GuiBanner: public GuiImage
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
//!Constructor
|
//!Constructor
|
||||||
//!\param tplfilepath Path of the tpl file
|
//!\param tplfilepath Path of the tpl file
|
||||||
GuiBanner( const char *tplfilepath );
|
GuiBanner(const char *tplfilepath);
|
||||||
//!Constructor
|
//!Constructor
|
||||||
//!\param mem Memory of the loaded tpl
|
//!\param mem Memory of the loaded tpl
|
||||||
//!\param len Filesize of the tpl
|
//!\param len Filesize of the tpl
|
||||||
//!\param w Width of the tpl
|
//!\param w Width of the tpl
|
||||||
//!\param h Height of the tpl
|
//!\param h Height of the tpl
|
||||||
GuiBanner( void *mem, u32 len, int w, int h );
|
GuiBanner(void *mem, u32 len, int w, int h);
|
||||||
//!Destructor
|
//!Destructor
|
||||||
~GuiBanner();
|
~GuiBanner();
|
||||||
void Draw();
|
void Draw();
|
||||||
|
@ -28,55 +28,54 @@
|
|||||||
#include "../ramdisk/ramdisk.h"
|
#include "../ramdisk/ramdisk.h"
|
||||||
#include "../listfiles.h"
|
#include "../listfiles.h"
|
||||||
|
|
||||||
u16 be16( const u8 *p )
|
u16 be16(const u8 *p)
|
||||||
{
|
{
|
||||||
return ( p[0] << 8 ) | p[1];
|
return (p[0] << 8) | p[1];
|
||||||
}
|
}
|
||||||
|
|
||||||
u32 be32( const u8 *p )
|
u32 be32(const u8 *p)
|
||||||
{
|
{
|
||||||
return ( p[0] << 24 ) | ( p[1] << 16 ) | ( p[2] << 8 ) | p[3];
|
return (p[0] << 24) | (p[1] << 16) | (p[2] << 8) | p[3];
|
||||||
}
|
}
|
||||||
|
|
||||||
u64 be64( const u8 *p )
|
u64 be64(const u8 *p)
|
||||||
{
|
{
|
||||||
return ( ( u64 )be32( p ) << 32 ) | be32( p + 4 );
|
return ((u64) be32(p) << 32) | be32(p + 4);
|
||||||
}
|
}
|
||||||
|
|
||||||
u64 be34( const u8 *p )
|
u64 be34(const u8 *p)
|
||||||
{
|
{
|
||||||
return 4 * ( u64 )be32( p );
|
return 4 * (u64) be32(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
void wbe16( u8 *p, u16 x )
|
void wbe16(u8 *p, u16 x)
|
||||||
{
|
{
|
||||||
p[0] = x >> 8;
|
p[0] = x >> 8;
|
||||||
p[1] = x;
|
p[1] = x;
|
||||||
}
|
}
|
||||||
|
|
||||||
void wbe32( u8 *p, u32 x )
|
void wbe32(u8 *p, u32 x)
|
||||||
{
|
{
|
||||||
wbe16( p, x >> 16 );
|
wbe16(p, x >> 16);
|
||||||
wbe16( p + 2, x );
|
wbe16(p + 2, x);
|
||||||
}
|
}
|
||||||
|
|
||||||
void wbe64( u8 *p, u64 x )
|
void wbe64(u8 *p, u64 x)
|
||||||
{
|
{
|
||||||
wbe32( p, x >> 32 );
|
wbe32(p, x >> 32);
|
||||||
wbe32( p + 4, x );
|
wbe32(p + 4, x);
|
||||||
}
|
}
|
||||||
|
|
||||||
void md5( u8 *data, u32 len, u8 *hash )
|
void md5(u8 *data, u32 len, u8 *hash)
|
||||||
{
|
{
|
||||||
MD5( hash, data, len );
|
MD5(hash, data, len);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
u8 zeroes[0x40];
|
u8 zeroes[0x40];
|
||||||
u32 imet; // "IMET"
|
u32 imet; // "IMET"
|
||||||
u8 zero_six_zero_three[8]; // fixed, unknown purpose
|
u8 zero_six_zero_three[8]; // fixed, unknown purpose
|
||||||
u32 sizes[3];
|
u32 sizes[3];
|
||||||
u32 flag1;
|
u32 flag1;
|
||||||
u16 name_jp[0x2a]; // might be empty
|
u16 name_jp[0x2a]; // might be empty
|
||||||
@ -93,12 +92,12 @@ typedef struct
|
|||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
u32 imd5_tag; // 0x494D4435 "IMD5";
|
u32 imd5_tag; // 0x494D4435 "IMD5";
|
||||||
u32 size; // size of the rest of part B, starting from next field.
|
u32 size; // size of the rest of part B, starting from next field.
|
||||||
u8 zeroes[8];
|
u8 zeroes[8];
|
||||||
u8 md5[16];
|
u8 md5[16];
|
||||||
u32 payload_tag; // 0x4C5A3737 "LZ77" if this is lz77
|
u32 payload_tag; // 0x4C5A3737 "LZ77" if this is lz77
|
||||||
u32 payload_data;
|
u32 payload_data;
|
||||||
} imd5_header_t;
|
} imd5_header_t;
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
@ -117,20 +116,20 @@ typedef struct
|
|||||||
u8 zeroes[16];
|
u8 zeroes[16];
|
||||||
} U8_archive_header;
|
} U8_archive_header;
|
||||||
|
|
||||||
static int write_file( void* data, size_t size, char* name )
|
static int write_file(void* data, size_t size, char* name)
|
||||||
{
|
{
|
||||||
size_t written = 0;
|
size_t written = 0;
|
||||||
FILE *out;
|
FILE *out;
|
||||||
out = fopen( name, "wb" );
|
out = fopen(name, "wb");
|
||||||
if ( out )
|
if (out)
|
||||||
{
|
{
|
||||||
written = fwrite( data, 1, size, out );
|
written = fwrite(data, 1, size, out);
|
||||||
fclose( out );
|
fclose(out);
|
||||||
}
|
}
|
||||||
return ( written == size ) ? 1 : -1;
|
return (written == size) ? 1 : -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
u8* decompress_lz77( u8 *data, size_t data_size, size_t* decompressed_size )
|
u8* decompress_lz77(u8 *data, size_t data_size, size_t* decompressed_size)
|
||||||
{
|
{
|
||||||
u8 *data_end;
|
u8 *data_end;
|
||||||
u8 *decompressed_data;
|
u8 *decompressed_data;
|
||||||
@ -145,43 +144,43 @@ u8* decompress_lz77( u8 *data, size_t data_size, size_t* decompressed_size )
|
|||||||
// Assume this for now and grow when needed
|
// Assume this for now and grow when needed
|
||||||
unpacked_size = data_size;
|
unpacked_size = data_size;
|
||||||
|
|
||||||
decompressed_data = malloc( unpacked_size );
|
decompressed_data = malloc(unpacked_size);
|
||||||
out_end = decompressed_data + unpacked_size;
|
out_end = decompressed_data + unpacked_size;
|
||||||
|
|
||||||
out_ptr = decompressed_data;
|
out_ptr = decompressed_data;
|
||||||
|
|
||||||
while ( in_ptr < data_end )
|
while (in_ptr < data_end)
|
||||||
{
|
{
|
||||||
int bit;
|
int bit;
|
||||||
u8 bitmask = *in_ptr;
|
u8 bitmask = *in_ptr;
|
||||||
|
|
||||||
in_ptr++;
|
in_ptr++;
|
||||||
for ( bit = 0x80; bit != 0; bit >>= 1 )
|
for (bit = 0x80; bit != 0; bit >>= 1)
|
||||||
{
|
{
|
||||||
if ( bitmask & bit )
|
if (bitmask & bit)
|
||||||
{
|
{
|
||||||
// Next section is compressed
|
// Next section is compressed
|
||||||
u8 rep_length;
|
u8 rep_length;
|
||||||
u16 rep_offset;
|
u16 rep_offset;
|
||||||
|
|
||||||
rep_length = ( *in_ptr >> 4 ) + 3;
|
rep_length = (*in_ptr >> 4) + 3;
|
||||||
rep_offset = *in_ptr & 0x0f;
|
rep_offset = *in_ptr & 0x0f;
|
||||||
in_ptr++;
|
in_ptr++;
|
||||||
rep_offset = *in_ptr | ( rep_offset << 8 );
|
rep_offset = *in_ptr | (rep_offset << 8);
|
||||||
in_ptr++;
|
in_ptr++;
|
||||||
if ( out_ptr - decompressed_data < rep_offset )
|
if (out_ptr - decompressed_data < rep_offset)
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
for ( ; rep_length > 0; rep_length-- )
|
for (; rep_length > 0; rep_length--)
|
||||||
{
|
{
|
||||||
*out_ptr = out_ptr[-rep_offset-1];
|
*out_ptr = out_ptr[-rep_offset - 1];
|
||||||
out_ptr++;
|
out_ptr++;
|
||||||
if ( out_ptr >= out_end )
|
if (out_ptr >= out_end)
|
||||||
{
|
{
|
||||||
// Need to grow buffer
|
// Need to grow buffer
|
||||||
decompressed_data = realloc( decompressed_data, unpacked_size * 2 );
|
decompressed_data = realloc(decompressed_data, unpacked_size * 2);
|
||||||
out_ptr = decompressed_data + unpacked_size;
|
out_ptr = decompressed_data + unpacked_size;
|
||||||
unpacked_size *= 2;
|
unpacked_size *= 2;
|
||||||
out_end = decompressed_data + unpacked_size;
|
out_end = decompressed_data + unpacked_size;
|
||||||
@ -193,10 +192,10 @@ u8* decompress_lz77( u8 *data, size_t data_size, size_t* decompressed_size )
|
|||||||
// Just copy byte
|
// Just copy byte
|
||||||
*out_ptr = *in_ptr;
|
*out_ptr = *in_ptr;
|
||||||
out_ptr++;
|
out_ptr++;
|
||||||
if ( out_ptr >= out_end )
|
if (out_ptr >= out_end)
|
||||||
{
|
{
|
||||||
// Need to grow buffer
|
// Need to grow buffer
|
||||||
decompressed_data = realloc( decompressed_data, unpacked_size * 2 );
|
decompressed_data = realloc(decompressed_data, unpacked_size * 2);
|
||||||
out_ptr = decompressed_data + unpacked_size;
|
out_ptr = decompressed_data + unpacked_size;
|
||||||
unpacked_size *= 2;
|
unpacked_size *= 2;
|
||||||
out_end = decompressed_data + unpacked_size;
|
out_end = decompressed_data + unpacked_size;
|
||||||
@ -206,58 +205,58 @@ u8* decompress_lz77( u8 *data, size_t data_size, size_t* decompressed_size )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
*decompressed_size = ( out_ptr - decompressed_data );
|
*decompressed_size = (out_ptr - decompressed_data);
|
||||||
return decompressed_data;
|
return decompressed_data;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int write_imd5_lz77( u8* data, size_t size, char* outname )
|
static int write_imd5_lz77(u8* data, size_t size, char* outname)
|
||||||
{
|
{
|
||||||
imd5_header_t* header = ( imd5_header_t* ) data;
|
imd5_header_t* header = (imd5_header_t*) data;
|
||||||
u32 tag;
|
u32 tag;
|
||||||
u32 size_in_imd5;
|
u32 size_in_imd5;
|
||||||
u8 md5_calc[16];
|
u8 md5_calc[16];
|
||||||
u8 *decompressed_data;
|
u8 *decompressed_data;
|
||||||
size_t decompressed_size;
|
size_t decompressed_size;
|
||||||
|
|
||||||
tag = be32( ( u8* ) & header->imd5_tag );
|
tag = be32((u8*) &header->imd5_tag);
|
||||||
if ( tag != 0x494D4435 )
|
if (tag != 0x494D4435)
|
||||||
{
|
{
|
||||||
return -4;
|
return -4;
|
||||||
}
|
}
|
||||||
|
|
||||||
md5( data + 32, size - 32, md5_calc );
|
md5(data + 32, size - 32, md5_calc);
|
||||||
if ( memcmp( &header->md5, md5_calc, 0x10 ) )
|
if (memcmp(&header->md5, md5_calc, 0x10))
|
||||||
{
|
{
|
||||||
return -5;
|
return -5;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_in_imd5 = be32( ( u8* ) & header->size );
|
size_in_imd5 = be32((u8*) &header->size);
|
||||||
if ( size_in_imd5 != size - 32 )
|
if (size_in_imd5 != size - 32)
|
||||||
{
|
{
|
||||||
return -6;
|
return -6;
|
||||||
}
|
}
|
||||||
|
|
||||||
tag = be32( ( u8* ) & header->payload_tag );
|
tag = be32((u8*) &header->payload_tag);
|
||||||
if ( tag == 0x4C5A3737 )
|
if (tag == 0x4C5A3737)
|
||||||
{
|
{
|
||||||
// "LZ77" - uncompress
|
// "LZ77" - uncompress
|
||||||
decompressed_data = decompress_lz77( data + sizeof( imd5_header_t ), size - sizeof( imd5_header_t ), &decompressed_size );
|
decompressed_data = decompress_lz77(data + sizeof(imd5_header_t), size - sizeof(imd5_header_t),
|
||||||
if ( decompressed_data == NULL )
|
&decompressed_size);
|
||||||
return -7;
|
if (decompressed_data == NULL) return -7;
|
||||||
write_file( decompressed_data, decompressed_size, outname );
|
write_file(decompressed_data, decompressed_size, outname);
|
||||||
//printf(", uncompressed %d bytes, md5 ok", decompressed_size);
|
//printf(", uncompressed %d bytes, md5 ok", decompressed_size);
|
||||||
|
|
||||||
free( decompressed_data );
|
free(decompressed_data);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
write_file( &header->payload_tag, size - 32, outname );
|
write_file(&header->payload_tag, size - 32, outname);
|
||||||
//printf(", md5 ok");
|
//printf(", md5 ok");
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int do_U8_archive( FILE *fp )
|
static int do_U8_archive(FILE *fp)
|
||||||
{
|
{
|
||||||
U8_archive_header header;
|
U8_archive_header header;
|
||||||
U8_node root_node;
|
U8_node root_node;
|
||||||
@ -272,42 +271,42 @@ static int do_U8_archive( FILE *fp )
|
|||||||
u16 dir_stack[16];
|
u16 dir_stack[16];
|
||||||
int dir_index = 0;
|
int dir_index = 0;
|
||||||
|
|
||||||
fread( &header, 1, sizeof header, fp );
|
fread(&header, 1, sizeof header, fp);
|
||||||
tag = be32( ( u8* ) & header.tag );
|
tag = be32((u8*) &header.tag);
|
||||||
if ( tag != 0x55AA382D )
|
if (tag != 0x55AA382D)
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
fread( &root_node, 1, sizeof( root_node ), fp );
|
fread(&root_node, 1, sizeof(root_node), fp);
|
||||||
num_nodes = be32( ( u8* ) & root_node.size ) - 1;
|
num_nodes = be32((u8*) &root_node.size) - 1;
|
||||||
//printf("Number of files: %d\n", num_nodes);
|
//printf("Number of files: %d\n", num_nodes);
|
||||||
|
|
||||||
nodes = malloc( sizeof( U8_node ) * ( num_nodes ) );
|
nodes = malloc(sizeof(U8_node) * (num_nodes));
|
||||||
fread( nodes, 1, num_nodes * sizeof( U8_node ), fp );
|
fread(nodes, 1, num_nodes * sizeof(U8_node), fp);
|
||||||
|
|
||||||
data_offset = be32( ( u8* ) & header.data_offset );
|
data_offset = be32((u8*) &header.data_offset);
|
||||||
rest_size = data_offset - sizeof( header ) - ( num_nodes + 1 ) * sizeof( U8_node );
|
rest_size = data_offset - sizeof(header) - (num_nodes + 1) * sizeof(U8_node);
|
||||||
|
|
||||||
string_table = malloc( rest_size );
|
string_table = malloc(rest_size);
|
||||||
fread( string_table, 1, rest_size, fp );
|
fread(string_table, 1, rest_size, fp);
|
||||||
current_offset = data_offset;
|
current_offset = data_offset;
|
||||||
|
|
||||||
for ( i = 0; i < num_nodes; i++ )
|
for (i = 0; i < num_nodes; i++)
|
||||||
{
|
{
|
||||||
U8_node* node = &nodes[i];
|
U8_node* node = &nodes[i];
|
||||||
u16 type = be16( ( u8* ) & node->type );
|
u16 type = be16((u8*) &node->type);
|
||||||
u16 name_offset = be16( ( u8* ) & node->name_offset );
|
u16 name_offset = be16((u8*) &node->name_offset);
|
||||||
u32 my_data_offset = be32( ( u8* ) & node->data_offset );
|
u32 my_data_offset = be32((u8*) &node->data_offset);
|
||||||
u32 size = be32( ( u8* ) & node->size );
|
u32 size = be32((u8*) &node->size);
|
||||||
char* name = ( char* ) & string_table[name_offset];
|
char* name = (char*) &string_table[name_offset];
|
||||||
u8* file_data;
|
u8* file_data;
|
||||||
|
|
||||||
if ( type == 0x0100 )
|
if (type == 0x0100)
|
||||||
{
|
{
|
||||||
// Directory
|
// Directory
|
||||||
mkdir( name, 0777 );
|
mkdir(name, 0777);
|
||||||
chdir( name );
|
chdir(name);
|
||||||
dir_stack[++dir_index] = size;
|
dir_stack[++dir_index] = size;
|
||||||
//printf("%*s%s/\n", dir_index, "", name);
|
//printf("%*s%s/\n", dir_index, "", name);
|
||||||
}
|
}
|
||||||
@ -316,59 +315,59 @@ static int do_U8_archive( FILE *fp )
|
|||||||
// Normal file
|
// Normal file
|
||||||
u8 padding[32];
|
u8 padding[32];
|
||||||
|
|
||||||
if ( type != 0x0000 )
|
if (type != 0x0000)
|
||||||
{
|
{
|
||||||
free( string_table );
|
free(string_table);
|
||||||
return -2;
|
return -2;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( current_offset < my_data_offset )
|
if (current_offset < my_data_offset)
|
||||||
{
|
{
|
||||||
int diff = my_data_offset - current_offset;
|
int diff = my_data_offset - current_offset;
|
||||||
|
|
||||||
if ( diff > 32 )
|
if (diff > 32)
|
||||||
{
|
{
|
||||||
free( string_table );
|
free(string_table);
|
||||||
return -3;
|
return -3;
|
||||||
}
|
}
|
||||||
fread( padding, 1, diff, fp );
|
fread(padding, 1, diff, fp);
|
||||||
current_offset += diff;
|
current_offset += diff;
|
||||||
}
|
}
|
||||||
|
|
||||||
file_data = malloc( size );
|
file_data = malloc(size);
|
||||||
fread( file_data, 1, size, fp );
|
fread(file_data, 1, size, fp);
|
||||||
//printf("%*s %s (%d bytes", dir_index, "", name, size);
|
//printf("%*s %s (%d bytes", dir_index, "", name, size);
|
||||||
int result;
|
int result;
|
||||||
result = write_imd5_lz77( file_data, size, name );
|
result = write_imd5_lz77(file_data, size, name);
|
||||||
if ( result < 0 )
|
if (result < 0)
|
||||||
{
|
{
|
||||||
free( string_table );
|
free(string_table);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
//printf(")\n");
|
//printf(")\n");
|
||||||
current_offset += size;
|
current_offset += size;
|
||||||
}
|
}
|
||||||
|
|
||||||
while ( dir_stack[dir_index] == i + 2 && dir_index > 0 )
|
while (dir_stack[dir_index] == i + 2 && dir_index > 0)
|
||||||
{
|
{
|
||||||
chdir( ".." );
|
chdir("..");
|
||||||
dir_index--;
|
dir_index--;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
free( string_table );
|
free(string_table);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void do_imet_header( FILE *fp )
|
static void do_imet_header(FILE *fp)
|
||||||
{
|
{
|
||||||
imet_data_t header;
|
imet_data_t header;
|
||||||
|
|
||||||
fread( &header, 1, sizeof header, fp );
|
fread(&header, 1, sizeof header, fp);
|
||||||
|
|
||||||
write_file( &header, sizeof( header ), "header.imet" );
|
write_file(&header, sizeof(header), "header.imet");
|
||||||
}
|
}
|
||||||
|
|
||||||
void do_U8_archivebanner( FILE *fp )
|
void do_U8_archivebanner(FILE *fp)
|
||||||
{
|
{
|
||||||
U8_archive_header header;
|
U8_archive_header header;
|
||||||
U8_node root_node;
|
U8_node root_node;
|
||||||
@ -382,42 +381,42 @@ void do_U8_archivebanner( FILE *fp )
|
|||||||
u16 dir_stack[16];
|
u16 dir_stack[16];
|
||||||
int dir_index = 0;
|
int dir_index = 0;
|
||||||
|
|
||||||
fread( &header, 1, sizeof header, fp );
|
fread(&header, 1, sizeof header, fp);
|
||||||
tag = be32( ( u8* ) & header.tag );
|
tag = be32((u8*) &header.tag);
|
||||||
if ( tag != 0x55AA382D )
|
if (tag != 0x55AA382D)
|
||||||
{
|
{
|
||||||
//printf("No U8 tag");
|
//printf("No U8 tag");
|
||||||
exit( 0 );
|
exit(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
fread( &root_node, 1, sizeof( root_node ), fp );
|
fread(&root_node, 1, sizeof(root_node), fp);
|
||||||
num_nodes = be32( ( u8* ) & root_node.size ) - 1;
|
num_nodes = be32((u8*) &root_node.size) - 1;
|
||||||
printf( "Number of files: %d\n", num_nodes );
|
printf("Number of files: %d\n", num_nodes);
|
||||||
|
|
||||||
nodes = malloc( sizeof( U8_node ) * ( num_nodes ) );
|
nodes = malloc(sizeof(U8_node) * (num_nodes));
|
||||||
fread( nodes, 1, num_nodes * sizeof( U8_node ), fp );
|
fread(nodes, 1, num_nodes * sizeof(U8_node), fp);
|
||||||
|
|
||||||
data_offset = be32( ( u8* ) & header.data_offset );
|
data_offset = be32((u8*) &header.data_offset);
|
||||||
rest_size = data_offset - sizeof( header ) - ( num_nodes + 1 ) * sizeof( U8_node );
|
rest_size = data_offset - sizeof(header) - (num_nodes + 1) * sizeof(U8_node);
|
||||||
|
|
||||||
string_table = malloc( rest_size );
|
string_table = malloc(rest_size);
|
||||||
fread( string_table, 1, rest_size, fp );
|
fread(string_table, 1, rest_size, fp);
|
||||||
|
|
||||||
for ( i = 0; i < num_nodes; i++ )
|
for (i = 0; i < num_nodes; i++)
|
||||||
{
|
{
|
||||||
U8_node* node = &nodes[i];
|
U8_node* node = &nodes[i];
|
||||||
u16 type = be16( ( u8* ) & node->type );
|
u16 type = be16((u8*) &node->type);
|
||||||
u16 name_offset = be16( ( u8* ) & node->name_offset );
|
u16 name_offset = be16((u8*) &node->name_offset);
|
||||||
u32 my_data_offset = be32( ( u8* ) & node->data_offset );
|
u32 my_data_offset = be32((u8*) &node->data_offset);
|
||||||
u32 size = be32( ( u8* ) & node->size );
|
u32 size = be32((u8*) &node->size);
|
||||||
char* name = ( char* ) & string_table[name_offset];
|
char* name = (char*) &string_table[name_offset];
|
||||||
u8* file_data;
|
u8* file_data;
|
||||||
|
|
||||||
if ( type == 0x0100 )
|
if (type == 0x0100)
|
||||||
{
|
{
|
||||||
// Directory
|
// Directory
|
||||||
mkdir( name, 0777 );
|
mkdir(name, 0777);
|
||||||
chdir( name );
|
chdir(name);
|
||||||
dir_stack[++dir_index] = size;
|
dir_stack[++dir_index] = size;
|
||||||
//printf("%*s%s/\n", dir_index, "", name);
|
//printf("%*s%s/\n", dir_index, "", name);
|
||||||
}
|
}
|
||||||
@ -425,56 +424,57 @@ void do_U8_archivebanner( FILE *fp )
|
|||||||
{
|
{
|
||||||
// Normal file
|
// Normal file
|
||||||
|
|
||||||
if ( type != 0x0000 )
|
if (type != 0x0000)
|
||||||
{
|
{
|
||||||
printf( "Unknown type" );
|
printf("Unknown type");
|
||||||
exit( 0 );
|
exit(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
fseek( fp, my_data_offset, SEEK_SET );
|
fseek(fp, my_data_offset, SEEK_SET);
|
||||||
file_data = malloc( size );
|
file_data = malloc(size);
|
||||||
fread( file_data, 1, size, fp );
|
fread(file_data, 1, size, fp);
|
||||||
write_file( file_data, size, name );
|
write_file(file_data, size, name);
|
||||||
free( file_data );
|
free(file_data);
|
||||||
//printf("%*s %s (%d bytes)\n", dir_index, "", name, size);
|
//printf("%*s %s (%d bytes)\n", dir_index, "", name, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
while ( dir_stack[dir_index] == i + 2 && dir_index > 0 )
|
while (dir_stack[dir_index] == i + 2 && dir_index > 0)
|
||||||
{
|
{
|
||||||
chdir( ".." );
|
chdir("..");
|
||||||
dir_index--;
|
dir_index--;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
free( string_table );
|
free(string_table);
|
||||||
}
|
}
|
||||||
|
|
||||||
int extractbnrfile( const char * filepath, const char * destpath )
|
int extractbnrfile(const char * filepath, const char * destpath)
|
||||||
{
|
{
|
||||||
int ret = -1;
|
int ret = -1;
|
||||||
FILE *fp = fopen( filepath, "rb" );
|
FILE *fp = fopen(filepath, "rb");
|
||||||
if ( fp )
|
if (fp)
|
||||||
{
|
{
|
||||||
subfoldercreate( destpath );
|
subfoldercreate(destpath);
|
||||||
chdir( destpath );
|
chdir(destpath);
|
||||||
|
|
||||||
do_imet_header( fp );
|
do_imet_header(fp);
|
||||||
ret = do_U8_archive( fp );
|
ret = do_U8_archive(fp);
|
||||||
|
|
||||||
fclose( fp );
|
fclose(fp);
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int unpackBin( const char * filename, const char * outdir )
|
int unpackBin(const char * filename, const char * outdir)
|
||||||
{
|
{
|
||||||
FILE *fp = fopen( filename, "rb" );;
|
FILE *fp = fopen(filename, "rb");
|
||||||
if ( fp )
|
;
|
||||||
|
if (fp)
|
||||||
{
|
{
|
||||||
subfoldercreate( outdir );
|
subfoldercreate(outdir);
|
||||||
chdir( outdir );
|
chdir(outdir);
|
||||||
|
|
||||||
do_U8_archivebanner( fp );
|
do_U8_archivebanner(fp);
|
||||||
fclose( fp );
|
fclose(fp);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
@ -483,81 +483,79 @@ int unpackBin( const char * filename, const char * outdir )
|
|||||||
#define TMP_PATH(s) "BANNER:/dump"s
|
#define TMP_PATH(s) "BANNER:/dump"s
|
||||||
//#define TMP_PATH(s) "SD:/dump"s
|
//#define TMP_PATH(s) "SD:/dump"s
|
||||||
|
|
||||||
int unpackBanner( const u8 *gameid, int what, const char *outdir )
|
int unpackBanner(const u8 *gameid, int what, const char *outdir)
|
||||||
{
|
{
|
||||||
|
|
||||||
char path[256];
|
char path[256];
|
||||||
if ( !ramdiskMount( "BANNER", NULL ) ) return -1;
|
if (!ramdiskMount("BANNER", NULL)) return -1;
|
||||||
|
|
||||||
subfoldercreate( TMP_PATH( "/" ) );
|
subfoldercreate(TMP_PATH( "/" ));
|
||||||
s32 ret = dump_banner( gameid, TMP_PATH( "/opening.bnr" ) );
|
s32 ret = dump_banner(gameid, TMP_PATH( "/opening.bnr" ));
|
||||||
if ( ret != 1 )
|
if (ret != 1)
|
||||||
{
|
{
|
||||||
ret = -1;
|
ret = -1;
|
||||||
goto error2;
|
goto error2;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = extractbnrfile( TMP_PATH( "/opening.bnr" ), TMP_PATH( "/" ) );
|
ret = extractbnrfile(TMP_PATH( "/opening.bnr" ), TMP_PATH( "/" ));
|
||||||
if ( ret != 0 )
|
if (ret != 0)
|
||||||
{
|
{
|
||||||
ret = -1;
|
ret = -1;
|
||||||
goto error2;
|
goto error2;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( what & UNPACK_BANNER_BIN )
|
if (what & UNPACK_BANNER_BIN)
|
||||||
{
|
{
|
||||||
snprintf( path, sizeof( path ), "%sbanner/", outdir );
|
snprintf(path, sizeof(path), "%sbanner/", outdir);
|
||||||
ret = unpackBin( TMP_PATH( "/meta/banner.bin" ), path );
|
ret = unpackBin(TMP_PATH( "/meta/banner.bin" ), path);
|
||||||
if ( ret != 1 )
|
if (ret != 1)
|
||||||
{
|
{
|
||||||
ret = -1;
|
ret = -1;
|
||||||
goto error2;
|
goto error2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ( what & UNPACK_ICON_BIN )
|
if (what & UNPACK_ICON_BIN)
|
||||||
{
|
{
|
||||||
snprintf( path, sizeof( path ), "%sicon/", outdir );
|
snprintf(path, sizeof(path), "%sicon/", outdir);
|
||||||
ret = unpackBin( TMP_PATH( "/meta/icon.bin" ), path );
|
ret = unpackBin(TMP_PATH( "/meta/icon.bin" ), path);
|
||||||
if ( ret != 1 )
|
if (ret != 1)
|
||||||
{
|
{
|
||||||
ret = -1;
|
ret = -1;
|
||||||
goto error2;
|
goto error2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ( what & UNPACK_SOUND_BIN )
|
if (what & UNPACK_SOUND_BIN)
|
||||||
{
|
{
|
||||||
snprintf( path, sizeof( path ), "%ssound.bin", outdir );
|
snprintf(path, sizeof(path), "%ssound.bin", outdir);
|
||||||
FILE *fp = fopen( TMP_PATH( "/meta/sound.bin" ), "rb" );
|
FILE *fp = fopen(TMP_PATH( "/meta/sound.bin" ), "rb");
|
||||||
if ( fp )
|
if (fp)
|
||||||
{
|
{
|
||||||
size_t size;
|
size_t size;
|
||||||
u8 *data;
|
u8 *data;
|
||||||
fseek( fp, 0, SEEK_END );
|
fseek(fp, 0, SEEK_END);
|
||||||
size = ftell( fp );
|
size = ftell(fp);
|
||||||
if ( !size )
|
if (!size)
|
||||||
{
|
{
|
||||||
ret = -1;
|
ret = -1;
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
fseek( fp, 0, SEEK_SET );
|
fseek(fp, 0, SEEK_SET);
|
||||||
data = ( u8 * )malloc( size );
|
data = (u8 *) malloc(size);
|
||||||
if ( !data )
|
if (!data)
|
||||||
{
|
{
|
||||||
ret = -1;
|
ret = -1;
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
if ( fread( data, 1, size, fp ) != size )
|
if (fread(data, 1, size, fp) != size)
|
||||||
{
|
{
|
||||||
ret = -1;
|
ret = -1;
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
ret = write_file( data, size, path );
|
ret = write_file(data, size, path);
|
||||||
}
|
}
|
||||||
error: fclose( fp );
|
error: fclose(fp);
|
||||||
}
|
}
|
||||||
ramdiskUnmount( "BANNER" );
|
ramdiskUnmount("BANNER");
|
||||||
error2:
|
error2: if (ret < 0) return ret;
|
||||||
if ( ret < 0 )
|
|
||||||
return ret;
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -14,31 +14,31 @@ extern "C"
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
/***********************************************************
|
/***********************************************************
|
||||||
* Error description:
|
* Error description:
|
||||||
* 0 Successfully extracted
|
* 0 Successfully extracted
|
||||||
* -1 No U8 tag
|
* -1 No U8 tag
|
||||||
* -2 Unknown type
|
* -2 Unknown type
|
||||||
* -3 Archive inconsistency, too much padding
|
* -3 Archive inconsistency, too much padding
|
||||||
* -4 No IMD5 tag
|
* -4 No IMD5 tag
|
||||||
* -5 MD5 mismatch
|
* -5 MD5 mismatch
|
||||||
* -6 Size mismatch
|
* -6 Size mismatch
|
||||||
* -7 Inconsistency in LZ77 encoding
|
* -7 Inconsistency in LZ77 encoding
|
||||||
************************************************************/
|
************************************************************/
|
||||||
|
|
||||||
//! Extract opening.bnr from filepath to destpath
|
//! Extract opening.bnr from filepath to destpath
|
||||||
//! Files extracted: banner.bin icon.bin and sound.bin
|
//! Files extracted: banner.bin icon.bin and sound.bin
|
||||||
int extractbnrfile( const char * filepath, const char * destpath );
|
int extractbnrfile(const char * filepath, const char * destpath);
|
||||||
int unpackBin( const char * filename, const char * outdir );
|
int unpackBin(const char * filename, const char * outdir);
|
||||||
#define UNPACK_BANNER_BIN 1 /* extract banner.bin to outdir/banner/ */
|
#define UNPACK_BANNER_BIN 1 /* extract banner.bin to outdir/banner/ */
|
||||||
#define UNPACK_ICON_BIN 2 /* extract icon.bin to outdir/icon/ */
|
#define UNPACK_ICON_BIN 2 /* extract icon.bin to outdir/icon/ */
|
||||||
#define UNPACK_SOUND_BIN 4 /* copies sound.bin to outdir/sound.bin */
|
#define UNPACK_SOUND_BIN 4 /* copies sound.bin to outdir/sound.bin */
|
||||||
#define UNPACK_ALL (UNPACK_SOUND_BIN | UNPACK_ICON_BIN | UNPACK_BANNER_BIN)
|
#define UNPACK_ALL (UNPACK_SOUND_BIN | UNPACK_ICON_BIN | UNPACK_BANNER_BIN)
|
||||||
int unpackBanner( const u8 * gameid, int what, const char *outdir );
|
int unpackBanner(const u8 * gameid, int what, const char *outdir);
|
||||||
//! Extract the lz77 compressed banner, icon and sound .bin
|
//! Extract the lz77 compressed banner, icon and sound .bin
|
||||||
u8* decompress_lz77( u8 *data, size_t data_size, size_t* decompressed_size );
|
u8* decompress_lz77(u8 *data, size_t data_size, size_t* decompressed_size);
|
||||||
|
|
||||||
u16 be16( const u8 *p );
|
u16 be16(const u8 *p);
|
||||||
u32 be32( const u8 *p );
|
u32 be32(const u8 *p);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
|
@ -10,104 +10,100 @@
|
|||||||
#include "language/gettext.h"
|
#include "language/gettext.h"
|
||||||
#include "bannersound.h"
|
#include "bannersound.h"
|
||||||
|
|
||||||
|
|
||||||
struct IMD5Header
|
struct IMD5Header
|
||||||
{
|
{
|
||||||
u32 fcc;
|
u32 fcc;
|
||||||
u32 filesize;
|
u32 filesize;
|
||||||
u8 zeroes[8];
|
u8 zeroes[8];
|
||||||
u8 crypto[16];
|
u8 crypto[16];
|
||||||
} __attribute__( ( packed ) );
|
}__attribute__( ( packed ) );
|
||||||
|
|
||||||
struct IMETHeader
|
struct IMETHeader
|
||||||
{
|
{
|
||||||
u8 zeroes[64];
|
u8 zeroes[64];
|
||||||
u32 fcc;
|
u32 fcc;
|
||||||
u8 unk[8];
|
u8 unk[8];
|
||||||
u32 iconSize;
|
u32 iconSize;
|
||||||
u32 bannerSize;
|
u32 bannerSize;
|
||||||
u32 soundSize;
|
u32 soundSize;
|
||||||
u32 flag1;
|
u32 flag1;
|
||||||
u8 names[7][84];
|
u8 names[7][84];
|
||||||
u8 zeroes_2[0x348];
|
u8 zeroes_2[0x348];
|
||||||
u8 crypto[16];
|
u8 crypto[16];
|
||||||
} __attribute__( ( packed ) );
|
}__attribute__( ( packed ) );
|
||||||
|
|
||||||
struct U8Header
|
struct U8Header
|
||||||
{
|
{
|
||||||
u32 fcc;
|
u32 fcc;
|
||||||
u32 rootNodeOffset;
|
u32 rootNodeOffset;
|
||||||
u32 headerSize;
|
u32 headerSize;
|
||||||
u32 dataOffset;
|
u32 dataOffset;
|
||||||
u8 zeroes[16];
|
u8 zeroes[16];
|
||||||
} __attribute__( ( packed ) );
|
}__attribute__( ( packed ) );
|
||||||
|
|
||||||
struct U8Entry
|
struct U8Entry
|
||||||
{
|
{
|
||||||
struct
|
struct
|
||||||
{
|
{
|
||||||
u32 fileType : 8;
|
u32 fileType :8;
|
||||||
u32 nameOffset : 24;
|
u32 nameOffset :24;
|
||||||
};
|
};
|
||||||
u32 fileOffset;
|
u32 fileOffset;
|
||||||
union
|
union
|
||||||
{
|
{
|
||||||
u32 fileLength;
|
u32 fileLength;
|
||||||
u32 numEntries;
|
u32 numEntries;
|
||||||
};
|
};
|
||||||
} __attribute__( ( packed ) );
|
}__attribute__( ( packed ) );
|
||||||
|
|
||||||
struct LZ77Info
|
struct LZ77Info
|
||||||
{
|
{
|
||||||
u16 length : 4;
|
u16 length :4;
|
||||||
u16 offset : 12;
|
u16 offset :12;
|
||||||
} __attribute__( ( packed ) );
|
}__attribute__( ( packed ) );
|
||||||
|
|
||||||
static char *u8Filename( const U8Entry *fst, int i )
|
static char *u8Filename(const U8Entry *fst, int i)
|
||||||
{
|
{
|
||||||
return ( char * )( fst + fst[0].numEntries ) + fst[i].nameOffset;
|
return (char *) (fst + fst[0].numEntries) + fst[i].nameOffset;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline u32 le32( u32 i )
|
inline u32 le32(u32 i)
|
||||||
{
|
{
|
||||||
return ( ( i & 0xFF ) << 24 ) | ( ( i & 0xFF00 ) << 8 ) | ( ( i & 0xFF0000 ) >> 8 ) | ( ( i & 0xFF000000 ) >> 24 );
|
return ((i & 0xFF) << 24) | ((i & 0xFF00) << 8) | ((i & 0xFF0000) >> 8) | ((i & 0xFF000000) >> 24);
|
||||||
}
|
}
|
||||||
|
|
||||||
inline u16 le16( u16 i )
|
inline u16 le16(u16 i)
|
||||||
{
|
{
|
||||||
return ( ( i & 0xFF ) << 8 ) | ( ( i & 0xFF00 ) >> 8 );
|
return ((i & 0xFF) << 8) | ((i & 0xFF00) >> 8);
|
||||||
}
|
}
|
||||||
|
|
||||||
static u8 *uncompressLZ77( const u8 *inBuf, u32 inLength, u32 &size )
|
static u8 *uncompressLZ77(const u8 *inBuf, u32 inLength, u32 &size)
|
||||||
{
|
{
|
||||||
u8 *buffer = NULL;
|
u8 *buffer = NULL;
|
||||||
if ( inLength <= 0x8 || *( ( const u32 * )inBuf ) != 0x4C5A3737 /*"LZ77"*/ || inBuf[4] != 0x10 )
|
if (inLength <= 0x8 || *((const u32 *) inBuf) != 0x4C5A3737 /*"LZ77"*/|| inBuf[4] != 0x10) return NULL;
|
||||||
return NULL;
|
u32 uncSize = le32(((const u32 *) inBuf)[1] << 8);
|
||||||
u32 uncSize = le32( ( ( const u32 * )inBuf )[1] << 8 );
|
|
||||||
|
|
||||||
const u8 *inBufEnd = inBuf + inLength;
|
const u8 *inBufEnd = inBuf + inLength;
|
||||||
inBuf += 8;
|
inBuf += 8;
|
||||||
buffer = new( std::nothrow ) u8[uncSize];
|
buffer = new (std::nothrow) u8[uncSize];
|
||||||
if ( !buffer )
|
if (!buffer) return buffer;
|
||||||
return buffer;
|
|
||||||
|
|
||||||
u8 *bufCur = buffer;
|
u8 *bufCur = buffer;
|
||||||
u8 *bufEnd = buffer + uncSize;
|
u8 *bufEnd = buffer + uncSize;
|
||||||
|
|
||||||
while ( bufCur < bufEnd && inBuf < inBufEnd )
|
while (bufCur < bufEnd && inBuf < inBufEnd)
|
||||||
{
|
{
|
||||||
u8 flags = *inBuf;
|
u8 flags = *inBuf;
|
||||||
++inBuf;
|
++inBuf;
|
||||||
for ( int i = 0; i < 8 && bufCur < bufEnd && inBuf < inBufEnd; ++i )
|
for (int i = 0; i < 8 && bufCur < bufEnd && inBuf < inBufEnd; ++i)
|
||||||
{
|
{
|
||||||
if ( ( flags & 0x80 ) != 0 )
|
if ((flags & 0x80) != 0)
|
||||||
{
|
{
|
||||||
const LZ77Info &info = *( const LZ77Info * )inBuf;
|
const LZ77Info &info = *(const LZ77Info *) inBuf;
|
||||||
inBuf += sizeof ( LZ77Info );
|
inBuf += sizeof(LZ77Info);
|
||||||
int length = info.length + 3;
|
int length = info.length + 3;
|
||||||
if ( bufCur - info.offset - 1 < buffer || bufCur + length > bufEnd )
|
if (bufCur - info.offset - 1 < buffer || bufCur + length > bufEnd) return buffer;
|
||||||
return buffer;
|
memcpy(bufCur, bufCur - info.offset - 1, length);
|
||||||
memcpy( bufCur, bufCur - info.offset - 1, length );
|
|
||||||
bufCur += length;
|
bufCur += length;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -123,87 +119,86 @@ static u8 *uncompressLZ77( const u8 *inBuf, u32 inLength, u32 &size )
|
|||||||
return buffer;
|
return buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
const u8 *LoadBannerSound( const u8 *discid, u32 *size )
|
const u8 *LoadBannerSound(const u8 *discid, u32 *size)
|
||||||
{
|
{
|
||||||
if ( !discid )
|
if (!discid) return NULL;
|
||||||
return NULL;
|
|
||||||
|
|
||||||
Disc_SetUSB( NULL );
|
Disc_SetUSB(NULL);
|
||||||
wbfs_disc_t *disc = WBFS_OpenDisc( ( u8 * ) discid );
|
wbfs_disc_t *disc = WBFS_OpenDisc((u8 *) discid);
|
||||||
if ( !disc )
|
if (!disc)
|
||||||
{
|
{
|
||||||
// WindowPrompt(tr("Can't find disc"), 0, tr("OK"));
|
// WindowPrompt(tr("Can't find disc"), 0, tr("OK"));
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
wiidisc_t *wdisc = wd_open_disc( ( int ( * )( void *, u32, u32, void * ) )wbfs_disc_read, disc );
|
wiidisc_t *wdisc = wd_open_disc((int(*)(void *, u32, u32, void *)) wbfs_disc_read, disc);
|
||||||
if ( !wdisc )
|
if (!wdisc)
|
||||||
{
|
{
|
||||||
//WindowPrompt(tr("Could not open Disc"), 0, tr("OK"));
|
//WindowPrompt(tr("Could not open Disc"), 0, tr("OK"));
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
u8 * opening_bnr = wd_extract_file( wdisc, ALL_PARTITIONS, ( char * ) "opening.bnr" );
|
u8 * opening_bnr = wd_extract_file(wdisc, ALL_PARTITIONS, (char *) "opening.bnr");
|
||||||
if ( !opening_bnr )
|
if (!opening_bnr)
|
||||||
{
|
{
|
||||||
//WindowPrompt(tr("ERROR"), tr("Failed to extract opening.bnr"), tr("OK"));
|
//WindowPrompt(tr("ERROR"), tr("Failed to extract opening.bnr"), tr("OK"));
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
wd_close_disc( wdisc );
|
wd_close_disc(wdisc);
|
||||||
WBFS_CloseDisc( disc );
|
WBFS_CloseDisc(disc);
|
||||||
|
|
||||||
const U8Entry *fst;
|
const U8Entry *fst;
|
||||||
|
|
||||||
const IMETHeader *imetHdr = ( IMETHeader * )opening_bnr;
|
const IMETHeader *imetHdr = (IMETHeader *) opening_bnr;
|
||||||
if ( imetHdr->fcc != 0x494D4554 /*"IMET"*/ )
|
if (imetHdr->fcc != 0x494D4554 /*"IMET"*/)
|
||||||
{
|
{
|
||||||
// WindowPrompt(tr("IMET Header wrong."), 0, tr("OK"));
|
// WindowPrompt(tr("IMET Header wrong."), 0, tr("OK"));
|
||||||
free( opening_bnr );
|
free(opening_bnr);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
const U8Header *bnrArcHdr = ( U8Header * )( imetHdr + 1 );
|
const U8Header *bnrArcHdr = (U8Header *) (imetHdr + 1);
|
||||||
|
|
||||||
fst = ( const U8Entry * )( ( ( const u8 * )bnrArcHdr ) + bnrArcHdr->rootNodeOffset );
|
fst = (const U8Entry *) (((const u8 *) bnrArcHdr) + bnrArcHdr->rootNodeOffset);
|
||||||
u32 i;
|
u32 i;
|
||||||
for ( i = 1; i < fst[0].numEntries; ++i )
|
for (i = 1; i < fst[0].numEntries; ++i)
|
||||||
if ( fst[i].fileType == 0 && strcasecmp( u8Filename( fst, i ), "sound.bin" ) == 0 )
|
if (fst[i].fileType == 0 && strcasecmp(u8Filename(fst, i), "sound.bin") == 0) break;
|
||||||
break;
|
if (i >= fst[0].numEntries)
|
||||||
if ( i >= fst[0].numEntries )
|
|
||||||
{
|
{
|
||||||
/* Not all games have a sound.bin and this message is annoying **/
|
/* Not all games have a sound.bin and this message is annoying **/
|
||||||
//WindowPrompt(tr("sound.bin not found."), 0, tr("OK"));
|
//WindowPrompt(tr("sound.bin not found."), 0, tr("OK"));
|
||||||
free( opening_bnr );
|
free(opening_bnr);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
const u8 *sound_bin = ( ( const u8 * )bnrArcHdr ) + fst[i].fileOffset;
|
const u8 *sound_bin = ((const u8 *) bnrArcHdr) + fst[i].fileOffset;
|
||||||
if ( ( ( IMD5Header * )sound_bin )->fcc != 0x494D4435 /*"IMD5"*/ )
|
if (((IMD5Header *) sound_bin)->fcc != 0x494D4435 /*"IMD5"*/)
|
||||||
{
|
{
|
||||||
// WindowPrompt(tr("IMD5 Header not right."), 0, tr("OK"));
|
// WindowPrompt(tr("IMD5 Header not right."), 0, tr("OK"));
|
||||||
free( opening_bnr );
|
free(opening_bnr);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
const u8 *soundChunk = sound_bin + sizeof ( IMD5Header );;
|
const u8 *soundChunk = sound_bin + sizeof(IMD5Header);
|
||||||
u32 soundChunkSize = fst[i].fileLength - sizeof ( IMD5Header );
|
;
|
||||||
|
u32 soundChunkSize = fst[i].fileLength - sizeof(IMD5Header);
|
||||||
|
|
||||||
if ( *( ( u32* )soundChunk ) == 0x4C5A3737 /*"LZ77"*/ )
|
if (*((u32*) soundChunk) == 0x4C5A3737 /*"LZ77"*/)
|
||||||
{
|
{
|
||||||
u32 uncSize = NULL;
|
u32 uncSize = NULL;
|
||||||
u8 * uncompressed_data = uncompressLZ77( soundChunk, soundChunkSize, uncSize );
|
u8 * uncompressed_data = uncompressLZ77(soundChunk, soundChunkSize, uncSize);
|
||||||
if ( !uncompressed_data )
|
if (!uncompressed_data)
|
||||||
{
|
{
|
||||||
// WindowPrompt(tr("Can't decompress LZ77"), 0, tr("OK"));
|
// WindowPrompt(tr("Can't decompress LZ77"), 0, tr("OK"));
|
||||||
free( opening_bnr );
|
free(opening_bnr);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
if ( size ) *size = uncSize;
|
if (size) *size = uncSize;
|
||||||
free( opening_bnr );
|
free(opening_bnr);
|
||||||
return uncompressed_data;
|
return uncompressed_data;
|
||||||
}
|
}
|
||||||
u8 *out = new( std::nothrow ) u8[soundChunkSize];
|
u8 *out = new (std::nothrow) u8[soundChunkSize];
|
||||||
if ( out )
|
if (out)
|
||||||
{
|
{
|
||||||
memcpy( out, soundChunk, soundChunkSize );
|
memcpy(out, soundChunk, soundChunkSize);
|
||||||
if ( size ) *size = soundChunkSize;
|
if (size) *size = soundChunkSize;
|
||||||
}
|
}
|
||||||
free( opening_bnr );
|
free(opening_bnr);
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
#ifndef BANNERSOUND_H
|
#ifndef BANNERSOUND_H
|
||||||
#define BANNERSOUND_H
|
#define BANNERSOUND_H
|
||||||
|
|
||||||
const u8 *LoadBannerSound( const u8 *discid, u32 *size );
|
const u8 *LoadBannerSound(const u8 *discid, u32 *size);
|
||||||
|
|
||||||
#endif /* BANNERSOUND_H */
|
#endif /* BANNERSOUND_H */
|
||||||
|
@ -23,148 +23,148 @@ extern GuiWindow * mainWindow;
|
|||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* CheatMenu
|
* CheatMenu
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
int CheatMenu( const char * gameID )
|
int CheatMenu(const char * gameID)
|
||||||
{
|
{
|
||||||
int choice = 0;
|
int choice = 0;
|
||||||
bool exit = false;
|
bool exit = false;
|
||||||
int ret = 1;
|
int ret = 1;
|
||||||
|
|
||||||
// because destroy GuiSound must wait while sound playing is finished, we use a global sound
|
// because destroy GuiSound must wait while sound playing is finished, we use a global sound
|
||||||
if ( !btnClick2 ) btnClick2 = new GuiSound( button_click2_pcm, button_click2_pcm_size, Settings.sfxvolume );
|
if (!btnClick2) btnClick2 = new GuiSound(button_click2_pcm, button_click2_pcm_size, Settings.sfxvolume);
|
||||||
// GuiSound btnClick(button_click2_pcm, button_click2_pcm_size, Settings.sfxvolume);
|
// GuiSound btnClick(button_click2_pcm, button_click2_pcm_size, Settings.sfxvolume);
|
||||||
|
|
||||||
char imgPath[100];
|
char imgPath[100];
|
||||||
snprintf( imgPath, sizeof( imgPath ), "%sbutton_dialogue_box.png", Settings.theme_path );
|
snprintf(imgPath, sizeof(imgPath), "%sbutton_dialogue_box.png", Settings.theme_path);
|
||||||
GuiImageData btnOutline( imgPath, button_dialogue_box_png );
|
GuiImageData btnOutline(imgPath, button_dialogue_box_png);
|
||||||
snprintf( imgPath, sizeof( imgPath ), "%ssettings_background.png", Settings.theme_path );
|
snprintf(imgPath, sizeof(imgPath), "%ssettings_background.png", Settings.theme_path);
|
||||||
GuiImageData settingsbg( imgPath, settings_background_png );
|
GuiImageData settingsbg(imgPath, settings_background_png);
|
||||||
GuiImage settingsbackground( &settingsbg );
|
GuiImage settingsbackground(&settingsbg);
|
||||||
|
|
||||||
GuiTrigger trigA;
|
GuiTrigger trigA;
|
||||||
trigA.SetSimpleTrigger( -1, WPAD_BUTTON_A | WPAD_CLASSIC_BUTTON_A, PAD_BUTTON_A );
|
trigA.SetSimpleTrigger(-1, WPAD_BUTTON_A | WPAD_CLASSIC_BUTTON_A, PAD_BUTTON_A);
|
||||||
GuiTrigger trigB;
|
GuiTrigger trigB;
|
||||||
trigB.SetButtonOnlyTrigger( -1, WPAD_BUTTON_B | WPAD_CLASSIC_BUTTON_B, PAD_BUTTON_B );
|
trigB.SetButtonOnlyTrigger(-1, WPAD_BUTTON_B | WPAD_CLASSIC_BUTTON_B, PAD_BUTTON_B);
|
||||||
|
|
||||||
GuiText backBtnTxt( tr( "Back" ) , 22, THEME.prompttext );
|
GuiText backBtnTxt(tr( "Back" ), 22, THEME.prompttext);
|
||||||
backBtnTxt.SetMaxWidth( btnOutline.GetWidth() - 30 );
|
backBtnTxt.SetMaxWidth(btnOutline.GetWidth() - 30);
|
||||||
GuiImage backBtnImg( &btnOutline );
|
GuiImage backBtnImg(&btnOutline);
|
||||||
GuiButton backBtn( &backBtnImg, &backBtnImg, 2, 3, -140, 400, &trigA, NULL, btnClick2, 1 );
|
GuiButton backBtn(&backBtnImg, &backBtnImg, 2, 3, -140, 400, &trigA, NULL, btnClick2, 1);
|
||||||
backBtn.SetLabel( &backBtnTxt );
|
backBtn.SetLabel(&backBtnTxt);
|
||||||
backBtn.SetTrigger( &trigB );
|
backBtn.SetTrigger(&trigB);
|
||||||
|
|
||||||
GuiText createBtnTxt( tr( "Create" ) , 22, THEME.prompttext );
|
GuiText createBtnTxt(tr( "Create" ), 22, THEME.prompttext);
|
||||||
createBtnTxt.SetMaxWidth( btnOutline.GetWidth() - 30 );
|
createBtnTxt.SetMaxWidth(btnOutline.GetWidth() - 30);
|
||||||
GuiImage createBtnImg( &btnOutline );
|
GuiImage createBtnImg(&btnOutline);
|
||||||
GuiButton createBtn( &createBtnImg, &createBtnImg, 2, 3, 160, 400, &trigA, NULL, btnClick2, 1 );
|
GuiButton createBtn(&createBtnImg, &createBtnImg, 2, 3, 160, 400, &trigA, NULL, btnClick2, 1);
|
||||||
createBtn.SetLabel( &createBtnTxt );
|
createBtn.SetLabel(&createBtnTxt);
|
||||||
|
|
||||||
char txtfilename[55];
|
char txtfilename[55];
|
||||||
snprintf( txtfilename, sizeof( txtfilename ), "%s%s.txt", Settings.TxtCheatcodespath, gameID );
|
snprintf(txtfilename, sizeof(txtfilename), "%s%s.txt", Settings.TxtCheatcodespath, gameID);
|
||||||
|
|
||||||
GCTCheats c;
|
GCTCheats c;
|
||||||
int check = c.openTxtfile( txtfilename );
|
int check = c.openTxtfile(txtfilename);
|
||||||
|
|
||||||
int download = 0;
|
int download = 0;
|
||||||
|
|
||||||
switch ( check )
|
switch (check)
|
||||||
{
|
{
|
||||||
case -1:
|
case -1:
|
||||||
WindowPrompt( tr( "Error" ), tr( "Cheatfile is blank" ), tr( "OK" ) );
|
WindowPrompt(tr( "Error" ), tr( "Cheatfile is blank" ), tr( "OK" ));
|
||||||
break;
|
break;
|
||||||
case 0:
|
case 0:
|
||||||
download = WindowPrompt( tr( "Error" ), tr( "No Cheatfile found" ), tr( "Download Now" ), tr( "Cancel" ) );
|
download = WindowPrompt(tr( "Error" ), tr( "No Cheatfile found" ), tr( "Download Now" ), tr( "Cancel" ));
|
||||||
if ( download == 1 )
|
if (download == 1)
|
||||||
{
|
{
|
||||||
download = CodeDownload( gameID );
|
download = CodeDownload(gameID);
|
||||||
if ( download < 0 || c.openTxtfile( txtfilename ) != 1 )
|
if (download < 0 || c.openTxtfile(txtfilename) != 1) break;
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
else
|
else break;
|
||||||
break;
|
|
||||||
case 1:
|
case 1:
|
||||||
int cntcheats = c.getCnt();
|
int cntcheats = c.getCnt();
|
||||||
customOptionList cheatslst( cntcheats );
|
customOptionList cheatslst(cntcheats);
|
||||||
GuiCustomOptionBrowser chtBrowser( 400, 280, &cheatslst, Settings.theme_path, "bg_options_settings.png", bg_options_settings_png, 1, 90 );
|
GuiCustomOptionBrowser chtBrowser(400, 280, &cheatslst, Settings.theme_path, "bg_options_settings.png",
|
||||||
chtBrowser.SetPosition( 0, 90 );
|
bg_options_settings_png, 1, 90);
|
||||||
chtBrowser.SetAlignment( ALIGN_CENTRE, ALIGN_TOP );
|
chtBrowser.SetPosition(0, 90);
|
||||||
chtBrowser.SetClickable( true );
|
chtBrowser.SetAlignment(ALIGN_CENTRE, ALIGN_TOP);
|
||||||
|
chtBrowser.SetClickable(true);
|
||||||
|
|
||||||
GuiText titleTxt( c.getGameName().c_str(), 28, ( GXColor ) {0, 0, 0, 255} );
|
GuiText titleTxt(c.getGameName().c_str(), 28, ( GXColor )
|
||||||
titleTxt.SetAlignment( ALIGN_CENTRE, ALIGN_TOP );
|
{ 0, 0, 0, 255});
|
||||||
titleTxt.SetMaxWidth( 350, SCROLL_HORIZONTAL );
|
titleTxt.SetAlignment(ALIGN_CENTRE, ALIGN_TOP);
|
||||||
titleTxt.SetPosition( 12, 40 );
|
titleTxt.SetMaxWidth(350, SCROLL_HORIZONTAL);
|
||||||
|
titleTxt.SetPosition(12, 40);
|
||||||
|
|
||||||
for ( int i = 0; i <= cntcheats; i++ )
|
for (int i = 0; i <= cntcheats; i++)
|
||||||
{
|
{
|
||||||
cheatslst.SetValue( i, "%s", c.getCheatName( i ).c_str() );
|
cheatslst.SetValue(i, "%s", c.getCheatName(i).c_str());
|
||||||
cheatslst.SetName( i, "OFF" );
|
cheatslst.SetName(i, "OFF");
|
||||||
}
|
}
|
||||||
|
|
||||||
HaltGui();
|
HaltGui();
|
||||||
GuiWindow w( screenwidth, screenheight );
|
GuiWindow w(screenwidth, screenheight);
|
||||||
w.Append( &settingsbackground );
|
w.Append(&settingsbackground);
|
||||||
w.Append( &titleTxt );
|
w.Append(&titleTxt);
|
||||||
w.Append( &backBtn );
|
w.Append(&backBtn);
|
||||||
w.Append( &createBtn );
|
w.Append(&createBtn);
|
||||||
w.Append( &chtBrowser );
|
w.Append(&chtBrowser);
|
||||||
mainWindow->SetState( STATE_DISABLED );
|
mainWindow->SetState(STATE_DISABLED);
|
||||||
mainWindow->ChangeFocus( &w );
|
mainWindow->ChangeFocus(&w);
|
||||||
mainWindow->Append( &w );
|
mainWindow->Append(&w);
|
||||||
ResumeGui();
|
ResumeGui();
|
||||||
|
|
||||||
while ( !exit )
|
while (!exit)
|
||||||
{
|
{
|
||||||
VIDEO_WaitVSync ();
|
VIDEO_WaitVSync();
|
||||||
|
|
||||||
ret = chtBrowser.GetClickedOption();
|
ret = chtBrowser.GetClickedOption();
|
||||||
if ( ret != -1 )
|
if (ret != -1)
|
||||||
{
|
{
|
||||||
const char *strCheck = cheatslst.GetName( ret );
|
const char *strCheck = cheatslst.GetName(ret);
|
||||||
if ( strncmp( strCheck, "ON", 2 ) == 0 )
|
if (strncmp(strCheck, "ON", 2) == 0)
|
||||||
{
|
{
|
||||||
cheatslst.SetName( ret, "%s", "OFF" );
|
cheatslst.SetName(ret, "%s", "OFF");
|
||||||
}
|
}
|
||||||
else if ( strncmp( strCheck, "OFF", 3 ) == 0 )
|
else if (strncmp(strCheck, "OFF", 3) == 0)
|
||||||
{
|
{
|
||||||
cheatslst.SetName( ret, "%s", "ON" );
|
cheatslst.SetName(ret, "%s", "ON");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( createBtn.GetState() == STATE_CLICKED )
|
if (createBtn.GetState() == STATE_CLICKED)
|
||||||
{
|
{
|
||||||
createBtn.ResetState();
|
createBtn.ResetState();
|
||||||
if ( cntcheats > 0 )
|
if (cntcheats > 0)
|
||||||
{
|
{
|
||||||
int selectednrs[30];
|
int selectednrs[30];
|
||||||
int x = 0;
|
int x = 0;
|
||||||
for ( int i = 0; i <= cntcheats; i++ )
|
for (int i = 0; i <= cntcheats; i++)
|
||||||
{
|
{
|
||||||
const char *strCheck = cheatslst.GetName( i );
|
const char *strCheck = cheatslst.GetName(i);
|
||||||
if ( strncmp( strCheck, "ON", 2 ) == 0 )
|
if (strncmp(strCheck, "ON", 2) == 0)
|
||||||
{
|
{
|
||||||
selectednrs[x] = i;
|
selectednrs[x] = i;
|
||||||
x++;
|
x++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ( x == 0 )
|
if (x == 0)
|
||||||
{
|
{
|
||||||
WindowPrompt( tr( "Error" ), tr( "No cheats were selected" ), tr( "OK" ) );
|
WindowPrompt(tr( "Error" ), tr( "No cheats were selected" ), tr( "OK" ));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
subfoldercreate( Settings.Cheatcodespath );
|
subfoldercreate(Settings.Cheatcodespath);
|
||||||
string chtpath = Settings.Cheatcodespath;
|
string chtpath = Settings.Cheatcodespath;
|
||||||
string gctfname = chtpath + c.getGameID() + ".gct";
|
string gctfname = chtpath + c.getGameID() + ".gct";
|
||||||
c.createGCT( selectednrs, x, gctfname.c_str() );
|
c.createGCT(selectednrs, x, gctfname.c_str());
|
||||||
WindowPrompt( tr( "GCT File created" ), NULL, tr( "OK" ) );
|
WindowPrompt(tr( "GCT File created" ), NULL, tr( "OK" ));
|
||||||
exit = true;
|
exit = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else WindowPrompt( tr( "Error" ), tr( "Could not create GCT file" ), tr( "OK" ) );
|
else WindowPrompt(tr( "Error" ), tr( "Could not create GCT file" ), tr( "OK" ));
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( backBtn.GetState() == STATE_CLICKED )
|
if (backBtn.GetState() == STATE_CLICKED)
|
||||||
{
|
{
|
||||||
backBtn.ResetState();
|
backBtn.ResetState();
|
||||||
exit = true;
|
exit = true;
|
||||||
@ -172,8 +172,8 @@ int CheatMenu( const char * gameID )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
HaltGui();
|
HaltGui();
|
||||||
mainWindow->SetState( STATE_DEFAULT );
|
mainWindow->SetState(STATE_DEFAULT);
|
||||||
mainWindow->Remove( &w );
|
mainWindow->Remove(&w);
|
||||||
ResumeGui();
|
ResumeGui();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -8,6 +8,6 @@
|
|||||||
#ifndef _CHEATMENU_H_
|
#ifndef _CHEATMENU_H_
|
||||||
#define _CHEATMENU_H_
|
#define _CHEATMENU_H_
|
||||||
|
|
||||||
int CheatMenu( const char * gameID );
|
int CheatMenu(const char * gameID);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -12,19 +12,19 @@
|
|||||||
|
|
||||||
#define ERRORRANGE "Error: CheatNr out of range"
|
#define ERRORRANGE "Error: CheatNr out of range"
|
||||||
|
|
||||||
GCTCheats::GCTCheats( void )
|
GCTCheats::GCTCheats(void)
|
||||||
{
|
{
|
||||||
iCntCheats = 0;
|
iCntCheats = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
GCTCheats::~GCTCheats( void )
|
GCTCheats::~GCTCheats(void)
|
||||||
{
|
{
|
||||||
|
|
||||||
string sGameID = "";
|
string sGameID = "";
|
||||||
string sGameTitle = "";
|
string sGameTitle = "";
|
||||||
/*string sCheatName[MAXCHEATS];
|
/*string sCheatName[MAXCHEATS];
|
||||||
string sCheats[MAXCHEATS];
|
string sCheats[MAXCHEATS];
|
||||||
string sCheatComment[MAXCHEATS];*/
|
string sCheatComment[MAXCHEATS];*/
|
||||||
}
|
}
|
||||||
|
|
||||||
int GCTCheats::getCnt()
|
int GCTCheats::getCnt()
|
||||||
@ -32,19 +32,19 @@ int GCTCheats::getCnt()
|
|||||||
return iCntCheats;
|
return iCntCheats;
|
||||||
}
|
}
|
||||||
|
|
||||||
string GCTCheats::getGameName( void )
|
string GCTCheats::getGameName(void)
|
||||||
{
|
{
|
||||||
return sGameTitle;
|
return sGameTitle;
|
||||||
}
|
}
|
||||||
|
|
||||||
string GCTCheats::getGameID( void )
|
string GCTCheats::getGameID(void)
|
||||||
{
|
{
|
||||||
return sGameID;
|
return sGameID;
|
||||||
}
|
}
|
||||||
|
|
||||||
string GCTCheats::getCheat( int nr )
|
string GCTCheats::getCheat(int nr)
|
||||||
{
|
{
|
||||||
if ( nr <= ( iCntCheats - 1 ) )
|
if (nr <= (iCntCheats - 1))
|
||||||
{
|
{
|
||||||
return sCheats[nr];
|
return sCheats[nr];
|
||||||
}
|
}
|
||||||
@ -54,9 +54,9 @@ string GCTCheats::getCheat( int nr )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
string GCTCheats::getCheatName( int nr )
|
string GCTCheats::getCheatName(int nr)
|
||||||
{
|
{
|
||||||
if ( nr <= ( iCntCheats - 1 ) )
|
if (nr <= (iCntCheats - 1))
|
||||||
{
|
{
|
||||||
return sCheatName[nr];
|
return sCheatName[nr];
|
||||||
}
|
}
|
||||||
@ -66,9 +66,9 @@ string GCTCheats::getCheatName( int nr )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
string GCTCheats::getCheatComment( int nr )
|
string GCTCheats::getCheatComment(int nr)
|
||||||
{
|
{
|
||||||
if ( nr <= ( iCntCheats - 1 ) )
|
if (nr <= (iCntCheats - 1))
|
||||||
{
|
{
|
||||||
return sCheatComment[nr];
|
return sCheatComment[nr];
|
||||||
}
|
}
|
||||||
@ -78,210 +78,200 @@ string GCTCheats::getCheatComment( int nr )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int GCTCheats::createGCT( int nr, const char * filename )
|
int GCTCheats::createGCT(int nr, const char * filename)
|
||||||
{
|
{
|
||||||
|
|
||||||
if ( nr == 0 )
|
if (nr == 0) return 0;
|
||||||
return 0;
|
|
||||||
|
|
||||||
ofstream filestr;
|
ofstream filestr;
|
||||||
filestr.open( filename );
|
filestr.open(filename);
|
||||||
|
|
||||||
if ( filestr.fail() )
|
if (filestr.fail()) return 0;
|
||||||
return 0;
|
|
||||||
|
|
||||||
//Header and Footer
|
//Header and Footer
|
||||||
char header[] = { 0x00, 0xd0, 0xc0, 0xde, 0x00, 0xd0, 0xc0, 0xde};
|
char header[] = { 0x00, 0xd0, 0xc0, 0xde, 0x00, 0xd0, 0xc0, 0xde };
|
||||||
char footer[] = { 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
char footer[] = { 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
|
||||||
|
|
||||||
string buf = getCheat( nr );
|
string buf = getCheat(nr);
|
||||||
filestr.write( header, sizeof( header ) );
|
filestr.write(header, sizeof(header));
|
||||||
|
|
||||||
int x = 0;
|
int x = 0;
|
||||||
long int li;
|
long int li;
|
||||||
int len = buf.size();
|
int len = buf.size();
|
||||||
|
|
||||||
while ( x < len )
|
while (x < len)
|
||||||
{
|
{
|
||||||
string temp = buf.substr( x, 2 );
|
string temp = buf.substr(x, 2);
|
||||||
li = strtol( temp.c_str(), NULL, 16 );
|
li = strtol(temp.c_str(), NULL, 16);
|
||||||
temp = li;
|
temp = li;
|
||||||
filestr.write( temp.c_str(), 1 );
|
filestr.write(temp.c_str(), 1);
|
||||||
x += 2;
|
x += 2;
|
||||||
}
|
}
|
||||||
filestr.write( footer, sizeof( footer ) );
|
filestr.write(footer, sizeof(footer));
|
||||||
|
|
||||||
filestr.close();
|
filestr.close();
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int GCTCheats::createGCT( const char * chtbuffer, const char * filename )
|
int GCTCheats::createGCT(const char * chtbuffer, const char * filename)
|
||||||
{
|
{
|
||||||
|
|
||||||
ofstream filestr;
|
ofstream filestr;
|
||||||
filestr.open( filename );
|
filestr.open(filename);
|
||||||
|
|
||||||
if ( filestr.fail() )
|
if (filestr.fail()) return 0;
|
||||||
return 0;
|
|
||||||
|
|
||||||
//Header and Footer
|
//Header and Footer
|
||||||
char header[] = { 0x00, 0xd0, 0xc0, 0xde, 0x00, 0xd0, 0xc0, 0xde};
|
char header[] = { 0x00, 0xd0, 0xc0, 0xde, 0x00, 0xd0, 0xc0, 0xde };
|
||||||
char footer[] = { 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
char footer[] = { 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
|
||||||
|
|
||||||
string buf = chtbuffer;
|
string buf = chtbuffer;
|
||||||
filestr.write( header, sizeof( header ) );
|
filestr.write(header, sizeof(header));
|
||||||
|
|
||||||
int x = 0;
|
int x = 0;
|
||||||
long int li;
|
long int li;
|
||||||
int len = buf.size();
|
int len = buf.size();
|
||||||
|
|
||||||
while ( x < len )
|
while (x < len)
|
||||||
{
|
{
|
||||||
string temp = buf.substr( x, 2 );
|
string temp = buf.substr(x, 2);
|
||||||
li = strtol( temp.c_str(), NULL, 16 );
|
li = strtol(temp.c_str(), NULL, 16);
|
||||||
temp = li;
|
temp = li;
|
||||||
filestr.write( temp.c_str(), 1 );
|
filestr.write(temp.c_str(), 1);
|
||||||
x += 2;
|
x += 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
filestr.write( footer, sizeof( footer ) );
|
filestr.write(footer, sizeof(footer));
|
||||||
|
|
||||||
filestr.close();
|
filestr.close();
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int GCTCheats::createGCT( int nr[], int cnt, const char * filename )
|
int GCTCheats::createGCT(int nr[], int cnt, const char * filename)
|
||||||
{
|
{
|
||||||
|
|
||||||
if ( cnt == 0 )
|
if (cnt == 0) return 0;
|
||||||
return 0;
|
|
||||||
|
|
||||||
ofstream filestr;
|
ofstream filestr;
|
||||||
filestr.open( filename );
|
filestr.open(filename);
|
||||||
|
|
||||||
if ( filestr.fail() )
|
if (filestr.fail()) return 0;
|
||||||
return 0;
|
|
||||||
|
|
||||||
//Header and Footer
|
//Header and Footer
|
||||||
char header[] = { 0x00, 0xd0, 0xc0, 0xde, 0x00, 0xd0, 0xc0, 0xde};
|
char header[] = { 0x00, 0xd0, 0xc0, 0xde, 0x00, 0xd0, 0xc0, 0xde };
|
||||||
char footer[] = { 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
char footer[] = { 0xF0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
|
||||||
|
|
||||||
filestr.write( header, sizeof( header ) );
|
filestr.write(header, sizeof(header));
|
||||||
|
|
||||||
int c = 0;
|
int c = 0;
|
||||||
while ( c != cnt )
|
while (c != cnt)
|
||||||
{
|
{
|
||||||
int actnr = nr[c];
|
int actnr = nr[c];
|
||||||
string buf = getCheat( actnr );
|
string buf = getCheat(actnr);
|
||||||
long int li;
|
long int li;
|
||||||
int len = buf.size();
|
int len = buf.size();
|
||||||
int x = 0;
|
int x = 0;
|
||||||
|
|
||||||
while ( x < len )
|
while (x < len)
|
||||||
{
|
{
|
||||||
string temp = buf.substr( x, 2 );
|
string temp = buf.substr(x, 2);
|
||||||
li = strtol( temp.c_str(), NULL, 16 );
|
li = strtol(temp.c_str(), NULL, 16);
|
||||||
temp = li;
|
temp = li;
|
||||||
filestr.write( temp.c_str(), 1 );
|
filestr.write(temp.c_str(), 1);
|
||||||
x += 2;
|
x += 2;
|
||||||
}
|
}
|
||||||
c++;
|
c++;
|
||||||
}
|
}
|
||||||
|
|
||||||
filestr.write( footer, sizeof( footer ) );
|
filestr.write(footer, sizeof(footer));
|
||||||
filestr.close();
|
filestr.close();
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int GCTCheats::openTxtfile( const char * filename )
|
int GCTCheats::openTxtfile(const char * filename)
|
||||||
{
|
{
|
||||||
ifstream filestr;
|
ifstream filestr;
|
||||||
int i = 0;
|
int i = 0;
|
||||||
string str;
|
string str;
|
||||||
filestr.open( filename );
|
filestr.open(filename);
|
||||||
|
|
||||||
if ( filestr.fail() )
|
if (filestr.fail()) return 0;
|
||||||
return 0;
|
|
||||||
|
|
||||||
filestr.seekg( 0, ios_base::end );
|
filestr.seekg(0, ios_base::end);
|
||||||
int size = filestr.tellg();
|
int size = filestr.tellg();
|
||||||
if ( size <= 0 ) return -1;
|
if (size <= 0) return -1;
|
||||||
filestr.seekg( 0, ios_base::beg );
|
filestr.seekg(0, ios_base::beg);
|
||||||
|
|
||||||
getline( filestr, sGameID );
|
getline(filestr, sGameID);
|
||||||
if ( sGameID[sGameID.length() - 1] == '\r' )
|
if (sGameID[sGameID.length() - 1] == '\r') sGameID.erase(sGameID.length() - 1);
|
||||||
sGameID.erase( sGameID.length() - 1 );
|
|
||||||
|
|
||||||
getline( filestr, sGameTitle );
|
getline(filestr, sGameTitle);
|
||||||
if ( sGameTitle[sGameTitle.length() - 1] == '\r' )
|
if (sGameTitle[sGameTitle.length() - 1] == '\r') sGameTitle.erase(sGameTitle.length() - 1);
|
||||||
sGameTitle.erase( sGameTitle.length() - 1 );
|
|
||||||
|
|
||||||
getline( filestr, sCheatName[i] ); // skip first line if file uses CRLF
|
getline(filestr, sCheatName[i]); // skip first line if file uses CRLF
|
||||||
if ( !sGameTitle[sGameTitle.length() - 1] == '\r' )
|
if (!sGameTitle[sGameTitle.length() - 1] == '\r') filestr.seekg(0, ios_base::beg);
|
||||||
filestr.seekg( 0, ios_base::beg );
|
|
||||||
|
|
||||||
while ( !filestr.eof() )
|
while (!filestr.eof())
|
||||||
{
|
{
|
||||||
getline( filestr, sCheatName[i] ); // '\n' delimiter by default
|
getline(filestr, sCheatName[i]); // '\n' delimiter by default
|
||||||
if ( sCheatName[i][sCheatName[i].length() - 1] == '\r' )
|
if (sCheatName[i][sCheatName[i].length() - 1] == '\r') sCheatName[i].erase(sCheatName[i].length() - 1);
|
||||||
sCheatName[i].erase( sCheatName[i].length() - 1 );
|
|
||||||
|
|
||||||
string cheatdata;
|
string cheatdata;
|
||||||
bool emptyline = false;
|
bool emptyline = false;
|
||||||
|
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
getline( filestr, str );
|
getline(filestr, str);
|
||||||
if ( str[str.length() - 1] == '\r' )
|
if (str[str.length() - 1] == '\r') str.erase(str.length() - 1);
|
||||||
str.erase( str.length() - 1 );
|
|
||||||
|
|
||||||
if ( str == "" || str[0] == '\r' || str[0] == '\n' )
|
if (str == "" || str[0] == '\r' || str[0] == '\n')
|
||||||
{
|
{
|
||||||
emptyline = true;
|
emptyline = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( IsCode( str ) )
|
if (IsCode(str))
|
||||||
{
|
{
|
||||||
// remove any garbage (comment) after code
|
// remove any garbage (comment) after code
|
||||||
while ( str.size() > 17 )
|
while (str.size() > 17)
|
||||||
{
|
{
|
||||||
str.erase( str.length() - 1 );
|
str.erase(str.length() - 1);
|
||||||
}
|
}
|
||||||
cheatdata.append( str );
|
cheatdata.append(str);
|
||||||
size_t found = cheatdata.find( ' ' );
|
size_t found = cheatdata.find(' ');
|
||||||
cheatdata.replace( found, 1, "" );
|
cheatdata.replace(found, 1, "");
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
//printf("%i",str.size());
|
//printf("%i",str.size());
|
||||||
sCheatComment[i] = str;
|
sCheatComment[i] = str;
|
||||||
}
|
}
|
||||||
if ( filestr.eof() ) break;
|
if (filestr.eof()) break;
|
||||||
|
|
||||||
}
|
} while (!emptyline);
|
||||||
while ( !emptyline );
|
|
||||||
|
|
||||||
sCheats[i] = cheatdata;
|
sCheats[i] = cheatdata;
|
||||||
i++;
|
i++;
|
||||||
if ( i == MAXCHEATS ) break;
|
if (i == MAXCHEATS) break;
|
||||||
}
|
}
|
||||||
iCntCheats = i;
|
iCntCheats = i;
|
||||||
filestr.close();
|
filestr.close();
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool GCTCheats::IsCode( const std::string& str )
|
bool GCTCheats::IsCode(const std::string& str)
|
||||||
{
|
{
|
||||||
if ( str[8] == ' ' && str.size() >= 17 )
|
if (str[8] == ' ' && str.size() >= 17)
|
||||||
{
|
{
|
||||||
// accept strings longer than 17 in case there is a comment on the same line as the code
|
// accept strings longer than 17 in case there is a comment on the same line as the code
|
||||||
char part1[9];
|
char part1[9];
|
||||||
char part2[9];
|
char part2[9];
|
||||||
snprintf( part1, sizeof( part1 ), "%c%c%c%c%c%c%c%c", str[0], str[1], str[2], str[3], str[4], str[5], str[6], str[7] );
|
snprintf(part1, sizeof(part1), "%c%c%c%c%c%c%c%c", str[0], str[1], str[2], str[3], str[4], str[5], str[6],
|
||||||
snprintf( part2, sizeof( part2 ), "%c%c%c%c%c%c%c%c", str[9], str[10], str[11], str[12], str[13], str[14], str[15], str[16] );
|
str[7]);
|
||||||
if ( ( strtok( part1, "0123456789ABCDEFabcdef" ) == NULL ) && ( strtok( part2, "0123456789ABCDEFabcdef" ) == NULL ) )
|
snprintf(part2, sizeof(part2), "%c%c%c%c%c%c%c%c", str[9], str[10], str[11], str[12], str[13], str[14],
|
||||||
|
str[15], str[16]);
|
||||||
|
if ((strtok(part1, "0123456789ABCDEFabcdef") == NULL) && (strtok(part2, "0123456789ABCDEFabcdef") == NULL))
|
||||||
{
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -26,50 +26,50 @@ class GCTCheats
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
//!Constructor
|
//!Constructor
|
||||||
GCTCheats( void );
|
GCTCheats(void);
|
||||||
//!Destructor
|
//!Destructor
|
||||||
~GCTCheats( void );
|
~GCTCheats(void);
|
||||||
//!Open txt file with cheats
|
//!Open txt file with cheats
|
||||||
//!\param filename name of TXT file
|
//!\param filename name of TXT file
|
||||||
//!\return error code
|
//!\return error code
|
||||||
int openTxtfile( const char * filename );
|
int openTxtfile(const char * filename);
|
||||||
//!Creates GCT file for one cheat
|
//!Creates GCT file for one cheat
|
||||||
//!\param nr selected Cheat Numbers
|
//!\param nr selected Cheat Numbers
|
||||||
//!\param filename name of GCT file
|
//!\param filename name of GCT file
|
||||||
//!\return error code
|
//!\return error code
|
||||||
int createGCT( int nr, const char * filename );
|
int createGCT(int nr, const char * filename);
|
||||||
//!Creates GCT file from a buffer
|
//!Creates GCT file from a buffer
|
||||||
//!\param chtbuffer buffer that holds the cheat data
|
//!\param chtbuffer buffer that holds the cheat data
|
||||||
//!\param filename name of GCT file
|
//!\param filename name of GCT file
|
||||||
//!\return error code
|
//!\return error code
|
||||||
int createGCT( const char * chtbuffer, const char * filename );
|
int createGCT(const char * chtbuffer, const char * filename);
|
||||||
//!Creates GCT file
|
//!Creates GCT file
|
||||||
//!\param nr[] array of selected Cheat Numbers
|
//!\param nr[] array of selected Cheat Numbers
|
||||||
//!\param cnt size of array
|
//!\param cnt size of array
|
||||||
//!\param filename name of GCT file
|
//!\param filename name of GCT file
|
||||||
//!\return error code
|
//!\return error code
|
||||||
int createGCT( int nr[], int cnt, const char * filename );
|
int createGCT(int nr[], int cnt, const char * filename);
|
||||||
//!Gets Count cheats
|
//!Gets Count cheats
|
||||||
//!\return Count cheats
|
//!\return Count cheats
|
||||||
int getCnt();
|
int getCnt();
|
||||||
//!Gets Game Name
|
//!Gets Game Name
|
||||||
//!\return Game Name
|
//!\return Game Name
|
||||||
string getGameName( void );
|
string getGameName(void);
|
||||||
//!Gets GameID
|
//!Gets GameID
|
||||||
//!\return GameID
|
//!\return GameID
|
||||||
string getGameID( void );
|
string getGameID(void);
|
||||||
//!Gets cheat data
|
//!Gets cheat data
|
||||||
//!\return cheat data
|
//!\return cheat data
|
||||||
string getCheat( int nr );
|
string getCheat(int nr);
|
||||||
//!Gets Cheat Name
|
//!Gets Cheat Name
|
||||||
//!\return Cheat Name
|
//!\return Cheat Name
|
||||||
string getCheatName( int nr );
|
string getCheatName(int nr);
|
||||||
//!Gets Cheat Comment
|
//!Gets Cheat Comment
|
||||||
//!\return Cheat Comment
|
//!\return Cheat Comment
|
||||||
string getCheatComment( int nr );
|
string getCheatComment(int nr);
|
||||||
//!Check if string is a code
|
//!Check if string is a code
|
||||||
//!\return true/false
|
//!\return true/false
|
||||||
bool IsCode( const std::string& s );
|
bool IsCode(const std::string& s);
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* _GCT_H */
|
#endif /* _GCT_H */
|
||||||
|
@ -32,43 +32,43 @@ extern sec_t _FAT_startSector;
|
|||||||
|
|
||||||
extern s32 wbfsDev;
|
extern s32 wbfsDev;
|
||||||
|
|
||||||
int fat_sd_mount = MOUNT_NONE;
|
int fat_sd_mount = MOUNT_NONE;
|
||||||
sec_t fat_sd_sec = 0; // u32
|
sec_t fat_sd_sec = 0; // u32
|
||||||
|
|
||||||
int fat_usb_mount = 0;
|
int fat_usb_mount = 0;
|
||||||
sec_t fat_usb_sec = 0;
|
sec_t fat_usb_sec = 0;
|
||||||
|
|
||||||
int fat_wbfs_mount = 0;
|
int fat_wbfs_mount = 0;
|
||||||
sec_t fat_wbfs_sec = 0;
|
sec_t fat_wbfs_sec = 0;
|
||||||
|
|
||||||
int fs_ntfs_mount = 0;
|
int fs_ntfs_mount = 0;
|
||||||
sec_t fs_ntfs_sec = 0;
|
sec_t fs_ntfs_sec = 0;
|
||||||
|
|
||||||
int USBDevice_Init()
|
int USBDevice_Init()
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_FAT
|
#ifdef DEBUG_FAT
|
||||||
gprintf( "USBDevice_Init()" );
|
gprintf("USBDevice_Init()");
|
||||||
#endif
|
#endif
|
||||||
//closing all open Files write back the cache and then shutdown em!
|
//closing all open Files write back the cache and then shutdown em!
|
||||||
fatUnmount( "USB:/" );
|
fatUnmount("USB:/");
|
||||||
//right now mounts first FAT-partition
|
//right now mounts first FAT-partition
|
||||||
|
|
||||||
//try first mount with cIOS
|
//try first mount with cIOS
|
||||||
// if (!fatMount("USB", &__io_wiiums, 0, CACHE, SECTORS)) {
|
// if (!fatMount("USB", &__io_wiiums, 0, CACHE, SECTORS)) {
|
||||||
// //try now mount with libogc
|
// //try now mount with libogc
|
||||||
if ( !fatMount( "USB", &__io_usbstorage2, 0, CACHE, SECTORS ) )
|
if (!fatMount("USB", &__io_usbstorage2, 0, CACHE, SECTORS))
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_FAT
|
#ifdef DEBUG_FAT
|
||||||
gprintf( ":-1\n" );
|
gprintf(":-1\n");
|
||||||
#endif
|
#endif
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
// }
|
// }
|
||||||
|
|
||||||
fat_usb_mount = 1;
|
fat_usb_mount = 1;
|
||||||
fat_usb_sec = _FAT_startSector;
|
fat_usb_sec = _FAT_startSector;
|
||||||
#ifdef DEBUG_FAT
|
#ifdef DEBUG_FAT
|
||||||
gprintf( ":0\n" );
|
gprintf(":0\n");
|
||||||
#endif
|
#endif
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -76,36 +76,36 @@ int USBDevice_Init()
|
|||||||
void USBDevice_deInit()
|
void USBDevice_deInit()
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_FAT
|
#ifdef DEBUG_FAT
|
||||||
gprintf( "USBDevice_deInit(): " );
|
gprintf("USBDevice_deInit(): ");
|
||||||
#endif
|
#endif
|
||||||
//closing all open Files write back the cache and then shutdown em!
|
//closing all open Files write back the cache and then shutdown em!
|
||||||
fatUnmount( "USB:/" );
|
fatUnmount("USB:/");
|
||||||
|
|
||||||
fat_usb_mount = 0;
|
fat_usb_mount = 0;
|
||||||
fat_usb_sec = 0;
|
fat_usb_sec = 0;
|
||||||
#ifdef DEBUG_FAT
|
#ifdef DEBUG_FAT
|
||||||
gprintf( "ok\n" );
|
gprintf("ok\n");
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
int WBFSDevice_Init( u32 sector )
|
int WBFSDevice_Init(u32 sector)
|
||||||
{
|
{
|
||||||
//closing all open Files write back the cache and then shutdown em!
|
//closing all open Files write back the cache and then shutdown em!
|
||||||
fatUnmount( "WBFS:/" );
|
fatUnmount("WBFS:/");
|
||||||
//right now mounts first FAT-partition
|
//right now mounts first FAT-partition
|
||||||
|
|
||||||
//try first mount with cIOS
|
//try first mount with cIOS
|
||||||
// if (!fatMount("WBFS", &__io_wiiums, 0, CACHE, SECTORS)) {
|
// if (!fatMount("WBFS", &__io_wiiums, 0, CACHE, SECTORS)) {
|
||||||
//try now mount with libogc
|
//try now mount with libogc
|
||||||
if ( !fatMount( "WBFS", &__io_usbstorage2, 0, CACHE, SECTORS ) )
|
if (!fatMount("WBFS", &__io_usbstorage2, 0, CACHE, SECTORS))
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
// }
|
// }
|
||||||
|
|
||||||
fat_wbfs_mount = 1;
|
fat_wbfs_mount = 1;
|
||||||
fat_wbfs_sec = _FAT_startSector;
|
fat_wbfs_sec = _FAT_startSector;
|
||||||
if ( sector && fat_wbfs_sec != sector )
|
if (sector && fat_wbfs_sec != sector)
|
||||||
{
|
{
|
||||||
// This is an error situation...actually, but is ignored in Config loader also
|
// This is an error situation...actually, but is ignored in Config loader also
|
||||||
// Should ask Oggzee about it...
|
// Should ask Oggzee about it...
|
||||||
@ -116,16 +116,15 @@ int WBFSDevice_Init( u32 sector )
|
|||||||
void WBFSDevice_deInit()
|
void WBFSDevice_deInit()
|
||||||
{
|
{
|
||||||
//closing all open Files write back the cache and then shutdown em!
|
//closing all open Files write back the cache and then shutdown em!
|
||||||
fatUnmount( "WBFS:/" );
|
fatUnmount("WBFS:/");
|
||||||
|
|
||||||
fat_wbfs_mount = 0;
|
fat_wbfs_mount = 0;
|
||||||
fat_wbfs_sec = 0;
|
fat_wbfs_sec = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int isInserted( const char *path )
|
int isInserted(const char *path)
|
||||||
{
|
{
|
||||||
if ( !strncmp( path, "USB:", 4 ) )
|
if (!strncmp(path, "USB:", 4)) return 1;
|
||||||
return 1;
|
|
||||||
|
|
||||||
return __io_sdhc.isInserted() || __io_wiisd.isInserted();
|
return __io_sdhc.isInserted() || __io_wiisd.isInserted();
|
||||||
}
|
}
|
||||||
@ -134,33 +133,33 @@ static u8 sdIsInited = 0;
|
|||||||
int SDCard_Init()
|
int SDCard_Init()
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_FAT
|
#ifdef DEBUG_FAT
|
||||||
gprintf( "SDCard_Init()" );
|
gprintf("SDCard_Init()");
|
||||||
#endif
|
#endif
|
||||||
//closing all open Files write back the cache and then shutdown em!
|
//closing all open Files write back the cache and then shutdown em!
|
||||||
fatUnmount( "SD:/" );
|
fatUnmount("SD:/");
|
||||||
//right now mounts first FAT-partition
|
//right now mounts first FAT-partition
|
||||||
if ( fatMount( "SD", &__io_wiisd, 0, CACHE, SECTORS ) )
|
if (fatMount("SD", &__io_wiisd, 0, CACHE, SECTORS))
|
||||||
{
|
{
|
||||||
fat_sd_mount = MOUNT_SD;
|
fat_sd_mount = MOUNT_SD;
|
||||||
fat_sd_sec = _FAT_startSector;
|
fat_sd_sec = _FAT_startSector;
|
||||||
sdIsInited = 1;
|
sdIsInited = 1;
|
||||||
#ifdef DEBUG_FAT
|
#ifdef DEBUG_FAT
|
||||||
gprintf( ":1\n" );
|
gprintf(":1\n");
|
||||||
#endif
|
#endif
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
else if ( fatMount( "SD", &__io_sdhc, 0, CACHE, SDHC_SECTOR_SIZE ) )
|
else if (fatMount("SD", &__io_sdhc, 0, CACHE, SDHC_SECTOR_SIZE))
|
||||||
{
|
{
|
||||||
fat_sd_mount = MOUNT_SDHC;
|
fat_sd_mount = MOUNT_SDHC;
|
||||||
fat_sd_sec = _FAT_startSector;
|
fat_sd_sec = _FAT_startSector;
|
||||||
sdIsInited = 1;
|
sdIsInited = 1;
|
||||||
#ifdef DEBUG_FAT
|
#ifdef DEBUG_FAT
|
||||||
gprintf( ":1\n" );
|
gprintf(":1\n");
|
||||||
#endif
|
#endif
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
#ifdef DEBUG_FAT
|
#ifdef DEBUG_FAT
|
||||||
gprintf( ":-1\n" );
|
gprintf(":-1\n");
|
||||||
#endif
|
#endif
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@ -168,29 +167,29 @@ int SDCard_Init()
|
|||||||
void SDCard_deInit()
|
void SDCard_deInit()
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_FAT
|
#ifdef DEBUG_FAT
|
||||||
gprintf( "SDCard_deInit( %d ): ", sdIsInited );
|
gprintf("SDCard_deInit( %d ): ", sdIsInited);
|
||||||
#endif
|
#endif
|
||||||
//closing all open Files write back the cache and then shutdown em!
|
//closing all open Files write back the cache and then shutdown em!
|
||||||
if( sdIsInited )
|
if (sdIsInited)
|
||||||
{
|
{
|
||||||
fatUnmount( "SD:/" );
|
fatUnmount("SD:/");
|
||||||
|
|
||||||
fat_sd_mount = MOUNT_NONE;
|
fat_sd_mount = MOUNT_NONE;
|
||||||
fat_sd_sec = 0;
|
fat_sd_sec = 0;
|
||||||
sdIsInited = 0;
|
sdIsInited = 0;
|
||||||
}
|
}
|
||||||
#ifdef DEBUG_FAT
|
#ifdef DEBUG_FAT
|
||||||
gprintf( "ok\n" );
|
gprintf("ok\n");
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void ntfsInit();
|
void ntfsInit();
|
||||||
|
|
||||||
s32 MountNTFS( u32 sector )
|
s32 MountNTFS(u32 sector)
|
||||||
{
|
{
|
||||||
s32 ret;
|
s32 ret;
|
||||||
|
|
||||||
if ( fs_ntfs_mount ) return 0;
|
if (fs_ntfs_mount) return 0;
|
||||||
//printf("mounting NTFS\n");
|
//printf("mounting NTFS\n");
|
||||||
//Wpad_WaitButtons();
|
//Wpad_WaitButtons();
|
||||||
_FAT_mem_init();
|
_FAT_mem_init();
|
||||||
@ -200,39 +199,39 @@ s32 MountNTFS( u32 sector )
|
|||||||
// ntfsInit resets locale settings
|
// ntfsInit resets locale settings
|
||||||
// which breaks unicode in console
|
// which breaks unicode in console
|
||||||
// so we change it back to C-UTF-8
|
// so we change it back to C-UTF-8
|
||||||
setlocale( LC_CTYPE, "C-UTF-8" );
|
setlocale(LC_CTYPE, "C-UTF-8");
|
||||||
setlocale( LC_MESSAGES, "C-UTF-8" );
|
setlocale(LC_MESSAGES, "C-UTF-8");
|
||||||
|
|
||||||
if ( wbfsDev == WBFS_DEVICE_USB )
|
if (wbfsDev == WBFS_DEVICE_USB)
|
||||||
{
|
{
|
||||||
/* Initialize WBFS interface */
|
/* Initialize WBFS interface */
|
||||||
// if (!__io_wiiums.startup()) {
|
// if (!__io_wiiums.startup()) {
|
||||||
ret = __io_usbstorage2.startup();
|
ret = __io_usbstorage2.startup();
|
||||||
if ( !ret )
|
if (!ret)
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
// }
|
// }
|
||||||
/* Mount device */
|
/* Mount device */
|
||||||
// if (!ntfsMount("NTFS", &__io_wiiums, sector, CACHE, SECTORS, NTFS_SHOW_HIDDEN_FILES | NTFS_RECOVER)) {
|
// if (!ntfsMount("NTFS", &__io_wiiums, sector, CACHE, SECTORS, NTFS_SHOW_HIDDEN_FILES | NTFS_RECOVER)) {
|
||||||
ret = ntfsMount( "NTFS", &__io_usbstorage2, sector, CACHE, SECTORS, NTFS_SHOW_HIDDEN_FILES | NTFS_RECOVER );
|
ret = ntfsMount("NTFS", &__io_usbstorage2, sector, CACHE, SECTORS, NTFS_SHOW_HIDDEN_FILES | NTFS_RECOVER);
|
||||||
if ( !ret )
|
if (!ret)
|
||||||
{
|
{
|
||||||
return -2;
|
return -2;
|
||||||
}
|
}
|
||||||
// }
|
// }
|
||||||
}
|
}
|
||||||
else if ( wbfsDev == WBFS_DEVICE_SDHC )
|
else if (wbfsDev == WBFS_DEVICE_SDHC)
|
||||||
{
|
{
|
||||||
if ( sdhc_mode_sd == 0 )
|
if (sdhc_mode_sd == 0)
|
||||||
{
|
{
|
||||||
ret = ntfsMount( "NTFS", &__io_sdhc, 0, CACHE, SECTORS, NTFS_SHOW_HIDDEN_FILES | NTFS_RECOVER );
|
ret = ntfsMount("NTFS", &__io_sdhc, 0, CACHE, SECTORS, NTFS_SHOW_HIDDEN_FILES | NTFS_RECOVER);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
ret = ntfsMount( "NTFS", &__io_sdhc, 0, CACHE, SECTORS_SD, NTFS_SHOW_HIDDEN_FILES | NTFS_RECOVER );
|
ret = ntfsMount("NTFS", &__io_sdhc, 0, CACHE, SECTORS_SD, NTFS_SHOW_HIDDEN_FILES | NTFS_RECOVER);
|
||||||
}
|
}
|
||||||
if ( !ret )
|
if (!ret)
|
||||||
{
|
{
|
||||||
return -5;
|
return -5;
|
||||||
}
|
}
|
||||||
@ -244,10 +243,10 @@ s32 MountNTFS( u32 sector )
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
s32 UnmountNTFS( void )
|
s32 UnmountNTFS(void)
|
||||||
{
|
{
|
||||||
/* Unmount device */
|
/* Unmount device */
|
||||||
ntfsUnmount( "NTFS:/", true );
|
ntfsUnmount("NTFS:/", true);
|
||||||
|
|
||||||
fs_ntfs_mount = 0;
|
fs_ntfs_mount = 0;
|
||||||
fs_ntfs_sec = 0;
|
fs_ntfs_sec = 0;
|
||||||
@ -259,17 +258,17 @@ void _FAT_mem_init()
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
void* _FAT_mem_allocate( size_t size )
|
void* _FAT_mem_allocate(size_t size)
|
||||||
{
|
{
|
||||||
return malloc( size );
|
return malloc(size);
|
||||||
}
|
}
|
||||||
|
|
||||||
void* _FAT_mem_align( size_t size )
|
void* _FAT_mem_align(size_t size)
|
||||||
{
|
{
|
||||||
return memalign( 32, size );
|
return memalign(32, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
void _FAT_mem_free( void *mem )
|
void _FAT_mem_free(void *mem)
|
||||||
{
|
{
|
||||||
free( mem );
|
free(mem);
|
||||||
}
|
}
|
||||||
|
@ -6,29 +6,29 @@ extern "C"
|
|||||||
{
|
{
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
extern int fat_sd_mount;
|
extern int fat_sd_mount;
|
||||||
extern sec_t fat_sd_sec;
|
extern sec_t fat_sd_sec;
|
||||||
extern int fat_usb_mount;
|
extern int fat_usb_mount;
|
||||||
extern sec_t fat_usb_sec;
|
extern sec_t fat_usb_sec;
|
||||||
extern int fat_wbfs_mount;
|
extern int fat_wbfs_mount;
|
||||||
extern sec_t fat_wbfs_sec;
|
extern sec_t fat_wbfs_sec;
|
||||||
|
|
||||||
int USBDevice_Init();
|
int USBDevice_Init();
|
||||||
void USBDevice_deInit();
|
void USBDevice_deInit();
|
||||||
int WBFSDevice_Init( u32 sector );
|
int WBFSDevice_Init(u32 sector);
|
||||||
void WBFSDevice_deInit();
|
void WBFSDevice_deInit();
|
||||||
int isInserted( const char *path );
|
int isInserted(const char *path);
|
||||||
int SDCard_Init();
|
int SDCard_Init();
|
||||||
void SDCard_deInit();
|
void SDCard_deInit();
|
||||||
|
|
||||||
s32 MountNTFS( u32 sector );
|
s32 MountNTFS(u32 sector);
|
||||||
s32 UnmountNTFS( void );
|
s32 UnmountNTFS(void);
|
||||||
|
|
||||||
extern int fat_usb_mount;
|
extern int fat_usb_mount;
|
||||||
extern sec_t fat_usb_sec;
|
extern sec_t fat_usb_sec;
|
||||||
extern int fat_wbfs_mount;
|
extern int fat_wbfs_mount;
|
||||||
extern sec_t fat_wbfs_sec;
|
extern sec_t fat_wbfs_sec;
|
||||||
extern int fs_ntfs_mount;
|
extern int fs_ntfs_mount;
|
||||||
extern sec_t fs_ntfs_sec;
|
extern sec_t fs_ntfs_sec;
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
|
@ -11,551 +11,551 @@
|
|||||||
|
|
||||||
#include <gccore.h>
|
#include <gccore.h>
|
||||||
|
|
||||||
extern const u8 font_ttf[];
|
extern const u8 font_ttf[];
|
||||||
extern const u32 font_ttf_size;
|
extern const u32 font_ttf_size;
|
||||||
|
|
||||||
extern const u8 clock_ttf[];
|
extern const u8 clock_ttf[];
|
||||||
extern const u32 clock_ttf_size;
|
extern const u32 clock_ttf_size;
|
||||||
|
|
||||||
extern const u8 closebutton_png[];
|
extern const u8 closebutton_png[];
|
||||||
extern const u32 closebutton_png_size;
|
extern const u32 closebutton_png_size;
|
||||||
|
|
||||||
extern const u8 gxlogo_png[];
|
extern const u8 gxlogo_png[];
|
||||||
extern const u32 gxlogo_png_size;
|
extern const u32 gxlogo_png_size;
|
||||||
|
|
||||||
extern const u8 sdcard_png[];
|
extern const u8 sdcard_png[];
|
||||||
extern const u32 sdcard_png_size;
|
extern const u32 sdcard_png_size;
|
||||||
|
|
||||||
extern const u8 sdcard_over_png[];
|
extern const u8 sdcard_over_png[];
|
||||||
extern const u32 sdcard_over_png_size;
|
extern const u32 sdcard_over_png_size;
|
||||||
|
|
||||||
extern const u8 Wifi_btn_png[];
|
extern const u8 Wifi_btn_png[];
|
||||||
extern const u32 Wifi_btn_png_size;
|
extern const u32 Wifi_btn_png_size;
|
||||||
|
|
||||||
extern const u8 Channel_btn_png[];
|
extern const u8 Channel_btn_png[];
|
||||||
extern const u32 Channel_btn_png_size;
|
extern const u32 Channel_btn_png_size;
|
||||||
|
|
||||||
extern const u8 wiimote_png[];
|
extern const u8 wiimote_png[];
|
||||||
extern const u32 wiimote_png_size;
|
extern const u32 wiimote_png_size;
|
||||||
|
|
||||||
extern const u8 bg_music_ogg[];
|
extern const u8 bg_music_ogg[];
|
||||||
extern const u32 bg_music_ogg_size;
|
extern const u32 bg_music_ogg_size;
|
||||||
|
|
||||||
extern const u8 credits_music_ogg[];
|
extern const u8 credits_music_ogg[];
|
||||||
extern const u32 credits_music_ogg_size;
|
extern const u32 credits_music_ogg_size;
|
||||||
|
|
||||||
extern const u8 gameinfo1_png[];
|
extern const u8 gameinfo1_png[];
|
||||||
extern const u32 gameinfo1_png_size;
|
extern const u32 gameinfo1_png_size;
|
||||||
|
|
||||||
extern const u8 gameinfo2_png[];
|
extern const u8 gameinfo2_png[];
|
||||||
extern const u32 gameinfo2_png_size;
|
extern const u32 gameinfo2_png_size;
|
||||||
|
|
||||||
extern const u8 gameinfo1a_png[];
|
extern const u8 gameinfo1a_png[];
|
||||||
extern const u32 gameinfo1a_png_size;
|
extern const u32 gameinfo1a_png_size;
|
||||||
|
|
||||||
extern const u8 gameinfo2a_png[];
|
extern const u8 gameinfo2a_png[];
|
||||||
extern const u32 gameinfo2a_png_size;
|
extern const u32 gameinfo2a_png_size;
|
||||||
|
|
||||||
extern const u8 menuin_ogg[];
|
extern const u8 menuin_ogg[];
|
||||||
extern const u32 menuin_ogg_size;
|
extern const u32 menuin_ogg_size;
|
||||||
|
|
||||||
extern const u8 menuout_ogg[];
|
extern const u8 menuout_ogg[];
|
||||||
extern const u32 menuout_ogg_size;
|
extern const u32 menuout_ogg_size;
|
||||||
|
|
||||||
extern const u8 success_ogg[];
|
extern const u8 success_ogg[];
|
||||||
extern const u32 success_ogg_size;
|
extern const u32 success_ogg_size;
|
||||||
|
|
||||||
extern const u8 credits_button_png[];
|
extern const u8 credits_button_png[];
|
||||||
extern const u32 credits_button_png_size;
|
extern const u32 credits_button_png_size;
|
||||||
|
|
||||||
extern const u8 credits_button_over_png[];
|
extern const u8 credits_button_over_png[];
|
||||||
extern const u32 credits_button_over_png_size;
|
extern const u32 credits_button_over_png_size;
|
||||||
|
|
||||||
extern const u8 button_over_pcm[];
|
extern const u8 button_over_pcm[];
|
||||||
extern const u32 button_over_pcm_size;
|
extern const u32 button_over_pcm_size;
|
||||||
|
|
||||||
extern const u8 button_click_pcm[];
|
extern const u8 button_click_pcm[];
|
||||||
extern const u32 button_click_pcm_size;
|
extern const u32 button_click_pcm_size;
|
||||||
|
|
||||||
extern const u8 button_click2_pcm[];
|
extern const u8 button_click2_pcm[];
|
||||||
extern const u32 button_click2_pcm_size;
|
extern const u32 button_click2_pcm_size;
|
||||||
|
|
||||||
extern const u8 tooltip_left_png[];
|
extern const u8 tooltip_left_png[];
|
||||||
extern const u32 tooltip_left_png_size;
|
extern const u32 tooltip_left_png_size;
|
||||||
|
|
||||||
extern const u8 tooltip_tile_png[];
|
extern const u8 tooltip_tile_png[];
|
||||||
extern const u32 tooltip_tile_png_size;
|
extern const u32 tooltip_tile_png_size;
|
||||||
|
|
||||||
extern const u8 tooltip_right_png[];
|
extern const u8 tooltip_right_png[];
|
||||||
extern const u32 tooltip_right_png_size;
|
extern const u32 tooltip_right_png_size;
|
||||||
|
|
||||||
extern const u8 startgame_arrow_left_png[];
|
extern const u8 startgame_arrow_left_png[];
|
||||||
extern const u32 startgame_arrow_left_png_size;
|
extern const u32 startgame_arrow_left_png_size;
|
||||||
|
|
||||||
extern const u8 startgame_arrow_right_png[];
|
extern const u8 startgame_arrow_right_png[];
|
||||||
extern const u32 startgame_arrow_right_png_size;
|
extern const u32 startgame_arrow_right_png_size;
|
||||||
|
|
||||||
extern const u8 credits_bg_png[];
|
extern const u8 credits_bg_png[];
|
||||||
extern const u32 credits_bg_png_size;
|
extern const u32 credits_bg_png_size;
|
||||||
|
|
||||||
extern const u8 little_star_png[];
|
extern const u8 little_star_png[];
|
||||||
extern const u32 little_star_png_size;
|
extern const u32 little_star_png_size;
|
||||||
|
|
||||||
extern const u8 background_png[];
|
extern const u8 background_png[];
|
||||||
extern const u32 background_png_size;
|
extern const u32 background_png_size;
|
||||||
|
|
||||||
extern const u8 wbackground_png[];
|
extern const u8 wbackground_png[];
|
||||||
extern const u32 wbackground_png_size;
|
extern const u32 wbackground_png_size;
|
||||||
|
|
||||||
extern const u8 bg_options_settings_png[];
|
extern const u8 bg_options_settings_png[];
|
||||||
extern const u32 bg_options_settings_png_size;
|
extern const u32 bg_options_settings_png_size;
|
||||||
|
|
||||||
extern const u8 settings_background_png[];
|
extern const u8 settings_background_png[];
|
||||||
extern const u32 settings_background_png_size;
|
extern const u32 settings_background_png_size;
|
||||||
|
|
||||||
extern const u8 bg_browser_png[];
|
extern const u8 bg_browser_png[];
|
||||||
extern const u32 bg_browser_png_size;
|
extern const u32 bg_browser_png_size;
|
||||||
|
|
||||||
extern const u8 icon_archives_png[];
|
extern const u8 icon_archives_png[];
|
||||||
extern const u32 icon_archives_png_size;
|
extern const u32 icon_archives_png_size;
|
||||||
|
|
||||||
//extern const u8 icon_default_png[];
|
//extern const u8 icon_default_png[];
|
||||||
//extern const u32 icon_default_png_size;
|
//extern const u32 icon_default_png_size;
|
||||||
|
|
||||||
extern const u8 icon_folder_png[];
|
extern const u8 icon_folder_png[];
|
||||||
extern const u32 icon_folder_png_size;
|
extern const u32 icon_folder_png_size;
|
||||||
/*
|
/*
|
||||||
extern const u8 icon_gfx_png[];
|
extern const u8 icon_gfx_png[];
|
||||||
extern const u32 icon_gfx_png_size;
|
extern const u32 icon_gfx_png_size;
|
||||||
|
|
||||||
extern const u8 icon_pls_png[];
|
extern const u8 icon_pls_png[];
|
||||||
extern const u32 icon_pls_png_size;
|
extern const u32 icon_pls_png_size;
|
||||||
|
|
||||||
extern const u8 icon_sfx_png[];
|
extern const u8 icon_sfx_png[];
|
||||||
extern const u32 icon_sfx_png_size;
|
extern const u32 icon_sfx_png_size;
|
||||||
|
|
||||||
extern const u8 icon_txt_png[];
|
extern const u8 icon_txt_png[];
|
||||||
extern const u32 icon_txt_png_size;
|
extern const u32 icon_txt_png_size;
|
||||||
|
|
||||||
extern const u8 icon_xml_png[];
|
extern const u8 icon_xml_png[];
|
||||||
extern const u32 icon_xml_png_size;
|
extern const u32 icon_xml_png_size;
|
||||||
*/
|
*/
|
||||||
extern const u8 bg_browser_selection_png[];
|
extern const u8 bg_browser_selection_png[];
|
||||||
extern const u32 bg_browser_selection_png_size;
|
extern const u32 bg_browser_selection_png_size;
|
||||||
|
|
||||||
extern const u8 addressbar_textbox_png[];
|
extern const u8 addressbar_textbox_png[];
|
||||||
extern const u32 addressbar_textbox_png_size;
|
extern const u32 addressbar_textbox_png_size;
|
||||||
|
|
||||||
extern const u8 browser_png[];
|
extern const u8 browser_png[];
|
||||||
extern const u32 browser_png_size;
|
extern const u32 browser_png_size;
|
||||||
|
|
||||||
extern const u8 browser_over_png[];
|
extern const u8 browser_over_png[];
|
||||||
extern const u32 browser_over_png_size;
|
extern const u32 browser_over_png_size;
|
||||||
|
|
||||||
extern const u8 nocover_png[];
|
extern const u8 nocover_png[];
|
||||||
extern const u32 nocover_png_size;
|
extern const u32 nocover_png_size;
|
||||||
|
|
||||||
extern const u8 nocoverFlat_png[];
|
extern const u8 nocoverFlat_png[];
|
||||||
extern const u32 nocoverFlat_png_size;
|
extern const u32 nocoverFlat_png_size;
|
||||||
|
|
||||||
extern const u8 nodisc_png[];
|
extern const u8 nodisc_png[];
|
||||||
extern const u32 nodisc_png_size;
|
extern const u32 nodisc_png_size;
|
||||||
|
|
||||||
extern const u8 theme_dialogue_box_png[];
|
extern const u8 theme_dialogue_box_png[];
|
||||||
extern const u32 theme_dialogue_box_png_size;
|
extern const u32 theme_dialogue_box_png_size;
|
||||||
|
|
||||||
extern const u8 button_install_png[];
|
extern const u8 button_install_png[];
|
||||||
extern const u32 button_install_png_size;
|
extern const u32 button_install_png_size;
|
||||||
|
|
||||||
extern const u8 button_install_over_png[];
|
extern const u8 button_install_over_png[];
|
||||||
extern const u32 button_install_over_png_size;
|
extern const u32 button_install_over_png_size;
|
||||||
|
|
||||||
extern const u8 dialogue_box_startgame_png[];
|
extern const u8 dialogue_box_startgame_png[];
|
||||||
extern const u32 dialogue_box_startgame_png_size;
|
extern const u32 dialogue_box_startgame_png_size;
|
||||||
|
|
||||||
extern const u8 wdialogue_box_startgame_png[];
|
extern const u8 wdialogue_box_startgame_png[];
|
||||||
extern const u32 wdialogue_box_startgame_png_size;
|
extern const u32 wdialogue_box_startgame_png_size;
|
||||||
|
|
||||||
extern const u8 button_dialogue_box_startgame_png[];
|
extern const u8 button_dialogue_box_startgame_png[];
|
||||||
extern const u32 button_dialogue_box_startgame_size;
|
extern const u32 button_dialogue_box_startgame_size;
|
||||||
|
|
||||||
extern const u8 button_dialogue_box_png[];
|
extern const u8 button_dialogue_box_png[];
|
||||||
extern const u32 button_dialogue_box_size;
|
extern const u32 button_dialogue_box_size;
|
||||||
|
|
||||||
extern const u8 keyboard_textbox_png[];
|
extern const u8 keyboard_textbox_png[];
|
||||||
extern const u32 keyboard_textbox_png_size;
|
extern const u32 keyboard_textbox_png_size;
|
||||||
|
|
||||||
extern const u8 keyboard_key_png[];
|
extern const u8 keyboard_key_png[];
|
||||||
extern const u32 keyboard_key_png_size;
|
extern const u32 keyboard_key_png_size;
|
||||||
|
|
||||||
extern const u8 keyboard_key_over_png[];
|
extern const u8 keyboard_key_over_png[];
|
||||||
extern const u32 keyboard_key_over_png_size;
|
extern const u32 keyboard_key_over_png_size;
|
||||||
|
|
||||||
extern const u8 keyboard_mediumkey_over_png[];
|
extern const u8 keyboard_mediumkey_over_png[];
|
||||||
extern const u32 keyboard_mediumkey_over_png_size;
|
extern const u32 keyboard_mediumkey_over_png_size;
|
||||||
|
|
||||||
extern const u8 keyboard_largekey_over_png[];
|
extern const u8 keyboard_largekey_over_png[];
|
||||||
extern const u32 keyboard_largekey_over_png_size;
|
extern const u32 keyboard_largekey_over_png_size;
|
||||||
|
|
||||||
extern const u8 keyboard_backspace_over_png[];
|
extern const u8 keyboard_backspace_over_png[];
|
||||||
extern const u32 keyboard_backspace_over_png_size;
|
extern const u32 keyboard_backspace_over_png_size;
|
||||||
|
|
||||||
extern const u8 keyboard_clear_over_png[];
|
extern const u8 keyboard_clear_over_png[];
|
||||||
extern const u32 keyboard_clear_over_png_size;
|
extern const u32 keyboard_clear_over_png_size;
|
||||||
|
|
||||||
extern const u8 menu_button_png[];
|
extern const u8 menu_button_png[];
|
||||||
extern const u32 menu_button_size;
|
extern const u32 menu_button_size;
|
||||||
|
|
||||||
extern const u8 menu_button_over_png[];
|
extern const u8 menu_button_over_png[];
|
||||||
extern const u32 menu_button_over_size;
|
extern const u32 menu_button_over_size;
|
||||||
|
|
||||||
extern const u8 settings_button_png[];
|
extern const u8 settings_button_png[];
|
||||||
extern const u32 settings_button_size;
|
extern const u32 settings_button_size;
|
||||||
|
|
||||||
extern const u8 settings_button_over_png[];
|
extern const u8 settings_button_over_png[];
|
||||||
extern const u32 settings_button_over_size;
|
extern const u32 settings_button_over_size;
|
||||||
|
|
||||||
extern const u8 settings_menu_button_png[];
|
extern const u8 settings_menu_button_png[];
|
||||||
extern const u32 settings_menu_button_size;
|
extern const u32 settings_menu_button_size;
|
||||||
|
|
||||||
extern const u8 wiimote_poweroff_png[];
|
extern const u8 wiimote_poweroff_png[];
|
||||||
extern const u32 wiimote_poweroff_png_size;
|
extern const u32 wiimote_poweroff_png_size;
|
||||||
|
|
||||||
extern const u8 dialogue_box_png[];
|
extern const u8 dialogue_box_png[];
|
||||||
extern const u32 dialogue_box_png_size;
|
extern const u32 dialogue_box_png_size;
|
||||||
|
|
||||||
extern const u8 theme_box_png[];
|
extern const u8 theme_box_png[];
|
||||||
extern const u32 theme_box_png_size;
|
extern const u32 theme_box_png_size;
|
||||||
|
|
||||||
extern const u8 wiimote_poweroff_over_png[];
|
extern const u8 wiimote_poweroff_over_png[];
|
||||||
extern const u32 wiimote_poweroff_over_png_size;
|
extern const u32 wiimote_poweroff_over_png_size;
|
||||||
|
|
||||||
extern const u8 bg_options_png[];
|
extern const u8 bg_options_png[];
|
||||||
extern const u32 bg_options_png_size;
|
extern const u32 bg_options_png_size;
|
||||||
|
|
||||||
extern const u8 bg_options_entry_png[];
|
extern const u8 bg_options_entry_png[];
|
||||||
extern const u32 bg_options_entry_png_size;
|
extern const u32 bg_options_entry_png_size;
|
||||||
|
|
||||||
extern const u8 scrollbar_png[];
|
extern const u8 scrollbar_png[];
|
||||||
extern const u32 scrollbar_png_size;
|
extern const u32 scrollbar_png_size;
|
||||||
|
|
||||||
extern const u8 scrollbar_arrowup_png[];
|
extern const u8 scrollbar_arrowup_png[];
|
||||||
extern const u32 scrollbar_arrowup_png_size;
|
extern const u32 scrollbar_arrowup_png_size;
|
||||||
|
|
||||||
extern const u8 scrollbar_arrowup_over_png[];
|
extern const u8 scrollbar_arrowup_over_png[];
|
||||||
extern const u32 scrollbar_arrowup_over_png_size;
|
extern const u32 scrollbar_arrowup_over_png_size;
|
||||||
|
|
||||||
extern const u8 scrollbar_arrowdown_png[];
|
extern const u8 scrollbar_arrowdown_png[];
|
||||||
extern const u32 scrollbar_arrowdown_png_size;
|
extern const u32 scrollbar_arrowdown_png_size;
|
||||||
|
|
||||||
extern const u8 scrollbar_arrowdown_over_png[];
|
extern const u8 scrollbar_arrowdown_over_png[];
|
||||||
extern const u32 scrollbar_arrowdown_over_png_size;
|
extern const u32 scrollbar_arrowdown_over_png_size;
|
||||||
|
|
||||||
extern const u8 scrollbar_box_png[];
|
extern const u8 scrollbar_box_png[];
|
||||||
extern const u32 scrollbar_box_png_size;
|
extern const u32 scrollbar_box_png_size;
|
||||||
|
|
||||||
extern const u8 scrollbar_box_over_png[];
|
extern const u8 scrollbar_box_over_png[];
|
||||||
extern const u32 scrollbar_box_over_png_size;
|
extern const u32 scrollbar_box_over_png_size;
|
||||||
|
|
||||||
extern const u8 progressbar_png[];
|
extern const u8 progressbar_png[];
|
||||||
extern const u32 progressbar_png_size;
|
extern const u32 progressbar_png_size;
|
||||||
|
|
||||||
extern const u8 progressbar_empty_png[];
|
extern const u8 progressbar_empty_png[];
|
||||||
extern const u32 progressbar_empty_png_size;
|
extern const u32 progressbar_empty_png_size;
|
||||||
|
|
||||||
extern const u8 progressbar_outline_png[];
|
extern const u8 progressbar_outline_png[];
|
||||||
extern const u32 progressbar_outline_png_size;
|
extern const u32 progressbar_outline_png_size;
|
||||||
|
|
||||||
extern const u8 player1_point_png[];
|
extern const u8 player1_point_png[];
|
||||||
extern const u32 player1_point_png_size;
|
extern const u32 player1_point_png_size;
|
||||||
|
|
||||||
extern const u8 player2_point_png[];
|
extern const u8 player2_point_png[];
|
||||||
extern const u32 player2_point_png_size;
|
extern const u32 player2_point_png_size;
|
||||||
|
|
||||||
extern const u8 player3_point_png[];
|
extern const u8 player3_point_png[];
|
||||||
extern const u32 player3_point_png_size;
|
extern const u32 player3_point_png_size;
|
||||||
|
|
||||||
extern const u8 player4_point_png[];
|
extern const u8 player4_point_png[];
|
||||||
extern const u32 player4_point_png_size;
|
extern const u32 player4_point_png_size;
|
||||||
|
|
||||||
extern const u8 rplayer1_point_png[];
|
extern const u8 rplayer1_point_png[];
|
||||||
extern const u32 rplayer1_point_png_size;
|
extern const u32 rplayer1_point_png_size;
|
||||||
|
|
||||||
extern const u8 rplayer2_point_png[];
|
extern const u8 rplayer2_point_png[];
|
||||||
extern const u32 rplayer2_point_png_size;
|
extern const u32 rplayer2_point_png_size;
|
||||||
|
|
||||||
extern const u8 rplayer3_point_png[];
|
extern const u8 rplayer3_point_png[];
|
||||||
extern const u32 rplayer3_point_png_size;
|
extern const u32 rplayer3_point_png_size;
|
||||||
|
|
||||||
extern const u8 rplayer4_point_png[];
|
extern const u8 rplayer4_point_png[];
|
||||||
extern const u32 rplayer4_point_png_size;
|
extern const u32 rplayer4_point_png_size;
|
||||||
|
|
||||||
extern const u8 battery_png[];
|
extern const u8 battery_png[];
|
||||||
extern const u32 battery_png_size;
|
extern const u32 battery_png_size;
|
||||||
|
|
||||||
extern const u8 battery_bar_png[];
|
extern const u8 battery_bar_png[];
|
||||||
extern const u32 battery_bar_png_size;
|
extern const u32 battery_bar_png_size;
|
||||||
|
|
||||||
extern const u8 battery_white_png[];
|
extern const u8 battery_white_png[];
|
||||||
extern const u32 battery_white_png_size;
|
extern const u32 battery_white_png_size;
|
||||||
|
|
||||||
extern const u8 battery_bar_white_png[];
|
extern const u8 battery_bar_white_png[];
|
||||||
extern const u32 battery_bar_white_png_size;
|
extern const u32 battery_bar_white_png_size;
|
||||||
|
|
||||||
extern const u8 battery_red_png[];
|
extern const u8 battery_red_png[];
|
||||||
extern const u32 battery_red_png_size;
|
extern const u32 battery_red_png_size;
|
||||||
|
|
||||||
extern const u8 battery_bar_red_png[];
|
extern const u8 battery_bar_red_png[];
|
||||||
extern const u32 battery_bar_red_png_size;
|
extern const u32 battery_bar_red_png_size;
|
||||||
|
|
||||||
extern const u8 arrow_next_png[];
|
extern const u8 arrow_next_png[];
|
||||||
extern const u32 arrow_next_png_size;
|
extern const u32 arrow_next_png_size;
|
||||||
|
|
||||||
extern const u8 arrow_previous_png[];
|
extern const u8 arrow_previous_png[];
|
||||||
extern const u32 arrow_previous_png_size;
|
extern const u32 arrow_previous_png_size;
|
||||||
|
|
||||||
extern const u8 mp3_pause_png[];
|
extern const u8 mp3_pause_png[];
|
||||||
extern const u32 mp3_pause_png_size;
|
extern const u32 mp3_pause_png_size;
|
||||||
|
|
||||||
extern const u8 exit_top_png[];
|
extern const u8 exit_top_png[];
|
||||||
extern const u32 exit_top_png_size;
|
extern const u32 exit_top_png_size;
|
||||||
|
|
||||||
extern const u8 exit_top_over_png[];
|
extern const u8 exit_top_over_png[];
|
||||||
extern const u32 exit_top_over_png_size;
|
extern const u32 exit_top_over_png_size;
|
||||||
|
|
||||||
extern const u8 exit_bottom_png[];
|
extern const u8 exit_bottom_png[];
|
||||||
extern const u32 exit_bottom_png_size;
|
extern const u32 exit_bottom_png_size;
|
||||||
|
|
||||||
extern const u8 exit_bottom_over_png[];
|
extern const u8 exit_bottom_over_png[];
|
||||||
extern const u32 exit_bottom_over_png_size;
|
extern const u32 exit_bottom_over_png_size;
|
||||||
|
|
||||||
extern const u8 exit_button_png[];
|
extern const u8 exit_button_png[];
|
||||||
extern const u32 exit_button_png_size;
|
extern const u32 exit_button_png_size;
|
||||||
|
|
||||||
extern const u8 mp3_stop_png[];
|
extern const u8 mp3_stop_png[];
|
||||||
extern const u32 mp3_stop_png_size;
|
extern const u32 mp3_stop_png_size;
|
||||||
|
|
||||||
extern const u8 favorite_png[];
|
extern const u8 favorite_png[];
|
||||||
extern const u32 favorite_png_size;
|
extern const u32 favorite_png_size;
|
||||||
|
|
||||||
extern const u8 not_favorite_png[];
|
extern const u8 not_favorite_png[];
|
||||||
extern const u32 not_favorite_png_size;
|
extern const u32 not_favorite_png_size;
|
||||||
|
|
||||||
extern const u8 favIcon_png[];
|
extern const u8 favIcon_png[];
|
||||||
extern const u32 favIcon_png_size;
|
extern const u32 favIcon_png_size;
|
||||||
|
|
||||||
extern const u8 searchIcon_png[];
|
extern const u8 searchIcon_png[];
|
||||||
extern const u32 searchIcon_png_size;
|
extern const u32 searchIcon_png_size;
|
||||||
|
|
||||||
extern const u8 abcIcon_png[];
|
extern const u8 abcIcon_png[];
|
||||||
extern const u32 abcIcon_png_size;
|
extern const u32 abcIcon_png_size;
|
||||||
|
|
||||||
extern const u8 rankIcon_png[];
|
extern const u8 rankIcon_png[];
|
||||||
extern const u32 rankIcon_png_size;
|
extern const u32 rankIcon_png_size;
|
||||||
|
|
||||||
extern const u8 playCountIcon_png[];
|
extern const u8 playCountIcon_png[];
|
||||||
extern const u32 playCountIcon_png_size;
|
extern const u32 playCountIcon_png_size;
|
||||||
|
|
||||||
extern const u8 arrangeList_png[];
|
extern const u8 arrangeList_png[];
|
||||||
extern const u32 arrangeList_png_size;
|
extern const u32 arrangeList_png_size;
|
||||||
|
|
||||||
extern const u8 arrangeGrid_png[];
|
extern const u8 arrangeGrid_png[];
|
||||||
extern const u32 arrangeGrid_png_size;
|
extern const u32 arrangeGrid_png_size;
|
||||||
|
|
||||||
extern const u8 arrangeCarousel_png[];
|
extern const u8 arrangeCarousel_png[];
|
||||||
extern const u32 arrangeCarousel_png_size;
|
extern const u32 arrangeCarousel_png_size;
|
||||||
|
|
||||||
extern const u8 settings_title_png[];
|
extern const u8 settings_title_png[];
|
||||||
extern const u32 settings_title_png_size;
|
extern const u32 settings_title_png_size;
|
||||||
|
|
||||||
extern const u8 settings_title_over_png[];
|
extern const u8 settings_title_over_png[];
|
||||||
extern const u32 settings_title_over_png_size;
|
extern const u32 settings_title_over_png_size;
|
||||||
|
|
||||||
extern const u8 pageindicator_png[];
|
extern const u8 pageindicator_png[];
|
||||||
extern const u32 pageindicator_png_size;
|
extern const u32 pageindicator_png_size;
|
||||||
|
|
||||||
extern const u8 Wiimote1_png[];
|
extern const u8 Wiimote1_png[];
|
||||||
extern const u32 Wiimote1_png_size;
|
extern const u32 Wiimote1_png_size;
|
||||||
|
|
||||||
extern const u8 Wiimote2_png[];
|
extern const u8 Wiimote2_png[];
|
||||||
extern const u32 Wiimote2_png_size;
|
extern const u32 Wiimote2_png_size;
|
||||||
|
|
||||||
extern const u8 Wiimote4_png[];
|
extern const u8 Wiimote4_png[];
|
||||||
extern const u32 Wiimote4_png_size;
|
extern const u32 Wiimote4_png_size;
|
||||||
|
|
||||||
extern const u8 wifi1_png[];
|
extern const u8 wifi1_png[];
|
||||||
extern const u32 wifi1_png_size;
|
extern const u32 wifi1_png_size;
|
||||||
|
|
||||||
extern const u8 wifi2_png[];
|
extern const u8 wifi2_png[];
|
||||||
extern const u32 wifi2png_size;
|
extern const u32 wifi2png_size;
|
||||||
|
|
||||||
extern const u8 wifi3_png[];
|
extern const u8 wifi3_png[];
|
||||||
extern const u32 wifi3_png_size;
|
extern const u32 wifi3_png_size;
|
||||||
|
|
||||||
extern const u8 wifi4_png[];
|
extern const u8 wifi4_png[];
|
||||||
extern const u32 wifi4_png_size;
|
extern const u32 wifi4_png_size;
|
||||||
|
|
||||||
//extern const u8 wifi6_png[];
|
//extern const u8 wifi6_png[];
|
||||||
//extern const u32 wifi6_png_size;
|
//extern const u32 wifi6_png_size;
|
||||||
|
|
||||||
extern const u8 wifi8_png[];
|
extern const u8 wifi8_png[];
|
||||||
extern const u32 wifi8_png_size;
|
extern const u32 wifi8_png_size;
|
||||||
|
|
||||||
extern const u8 wifi12_png[];
|
extern const u8 wifi12_png[];
|
||||||
extern const u32 wifi12_png_size;
|
extern const u32 wifi12_png_size;
|
||||||
|
|
||||||
extern const u8 wifi16_png[];
|
extern const u8 wifi16_png[];
|
||||||
extern const u32 wifi16_png_size;
|
extern const u32 wifi16_png_size;
|
||||||
|
|
||||||
extern const u8 wifi32_png[];
|
extern const u8 wifi32_png[];
|
||||||
extern const u32 wifi32_png_size;
|
extern const u32 wifi32_png_size;
|
||||||
|
|
||||||
extern const u8 norating_png[];
|
extern const u8 norating_png[];
|
||||||
extern const u32 norating_png_size;
|
extern const u32 norating_png_size;
|
||||||
|
|
||||||
extern const u8 guitar_png[];
|
extern const u8 guitar_png[];
|
||||||
extern const u32 guitar_png_size;
|
extern const u32 guitar_png_size;
|
||||||
extern const u8 guitarR_png[];
|
extern const u8 guitarR_png[];
|
||||||
extern const u32 guitarR_png_size;
|
extern const u32 guitarR_png_size;
|
||||||
|
|
||||||
extern const u8 microphone_png[];
|
extern const u8 microphone_png[];
|
||||||
extern const u32 microphone_png_size;
|
extern const u32 microphone_png_size;
|
||||||
extern const u8 microphoneR_png[];
|
extern const u8 microphoneR_png[];
|
||||||
extern const u32 microphoneR_png_size;
|
extern const u32 microphoneR_png_size;
|
||||||
|
|
||||||
extern const u8 gcncontroller_png[];
|
extern const u8 gcncontroller_png[];
|
||||||
extern const u32 gcncontroller_png_size;
|
extern const u32 gcncontroller_png_size;
|
||||||
extern const u8 gcncontrollerR_png[];
|
extern const u8 gcncontrollerR_png[];
|
||||||
extern const u32 gcncontrollerR_png_size;
|
extern const u32 gcncontrollerR_png_size;
|
||||||
|
|
||||||
extern const u8 classiccontroller_png[];
|
extern const u8 classiccontroller_png[];
|
||||||
extern const u32 classiccontroller_png_size;
|
extern const u32 classiccontroller_png_size;
|
||||||
extern const u8 classiccontrollerR_png[];
|
extern const u8 classiccontrollerR_png[];
|
||||||
extern const u32 classiccontrollerR_png_size;
|
extern const u32 classiccontrollerR_png_size;
|
||||||
|
|
||||||
extern const u8 nunchuk_png[];
|
extern const u8 nunchuk_png[];
|
||||||
extern const u32 nunchuk_png_size;
|
extern const u32 nunchuk_png_size;
|
||||||
extern const u8 nunchukR_png[];
|
extern const u8 nunchukR_png[];
|
||||||
extern const u32 nunchukR_png_size;
|
extern const u32 nunchukR_png_size;
|
||||||
|
|
||||||
extern const u8 dancepadR_png[];
|
extern const u8 dancepadR_png[];
|
||||||
extern const u32 dancepadR_size;
|
extern const u32 dancepadR_size;
|
||||||
extern const u8 dancepad_png[];
|
extern const u8 dancepad_png[];
|
||||||
extern const u32 dancepad_png_size;
|
extern const u32 dancepad_png_size;
|
||||||
|
|
||||||
extern const u8 balanceboard_png[];
|
extern const u8 balanceboard_png[];
|
||||||
extern const u32 balanceboard_png_size;
|
extern const u32 balanceboard_png_size;
|
||||||
extern const u8 balanceboardR_png[];
|
extern const u8 balanceboardR_png[];
|
||||||
extern const u32 balanceboardR_png_size;
|
extern const u32 balanceboardR_png_size;
|
||||||
|
|
||||||
extern const u8 drums_png[];
|
extern const u8 drums_png[];
|
||||||
extern const u32 drums_png_size;
|
extern const u32 drums_png_size;
|
||||||
extern const u8 drumsR_png[];
|
extern const u8 drumsR_png[];
|
||||||
extern const u32 drumsR_png_size;
|
extern const u32 drumsR_png_size;
|
||||||
|
|
||||||
extern const u8 motionplus_png[];
|
extern const u8 motionplus_png[];
|
||||||
extern const u32 motionplus_png_size;
|
extern const u32 motionplus_png_size;
|
||||||
extern const u8 motionplusR_png[];
|
extern const u8 motionplusR_png[];
|
||||||
extern const u32 motionplusR_png_size;
|
extern const u32 motionplusR_png_size;
|
||||||
|
|
||||||
extern const u8 wheel_png[];
|
extern const u8 wheel_png[];
|
||||||
extern const u32 wheel_png_size;
|
extern const u32 wheel_png_size;
|
||||||
extern const u8 wheelR_png[];
|
extern const u8 wheelR_png[];
|
||||||
extern const u32 wheelR_png_size;
|
extern const u32 wheelR_png_size;
|
||||||
|
|
||||||
extern const u8 zapper_png[];
|
extern const u8 zapper_png[];
|
||||||
extern const u32 zapper_png_size;
|
extern const u32 zapper_png_size;
|
||||||
extern const u8 zapperR_png[];
|
extern const u8 zapperR_png[];
|
||||||
extern const u32 zapperR_png_size;
|
extern const u32 zapperR_png_size;
|
||||||
|
|
||||||
extern const u8 wiispeak_png[];
|
extern const u8 wiispeak_png[];
|
||||||
extern const u32 wiispeak_png_size;
|
extern const u32 wiispeak_png_size;
|
||||||
extern const u8 wiispeakR_png[];
|
extern const u8 wiispeakR_png[];
|
||||||
extern const u32 wiispeakR_png_size;
|
extern const u32 wiispeakR_png_size;
|
||||||
|
|
||||||
extern const u8 nintendods_png[];
|
extern const u8 nintendods_png[];
|
||||||
extern const u32 nintendods_png_size;
|
extern const u32 nintendods_png_size;
|
||||||
extern const u8 nintendodsR_png[];
|
extern const u8 nintendodsR_png[];
|
||||||
extern const u32 nintendodsR_png_size;
|
extern const u32 nintendodsR_png_size;
|
||||||
/*
|
/*
|
||||||
extern const u8 vitalitysensor_png[];
|
extern const u8 vitalitysensor_png[];
|
||||||
extern const u32 vitalitysensor_png_size;
|
extern const u32 vitalitysensor_png_size;
|
||||||
extern const u8 vitalitysensor_png[];
|
extern const u8 vitalitysensor_png[];
|
||||||
extern const u32 vitalitysensorR_png_size;
|
extern const u32 vitalitysensorR_png_size;
|
||||||
*/
|
*/
|
||||||
extern const u8 esrb_ec_png[];
|
extern const u8 esrb_ec_png[];
|
||||||
extern const u32 esrb_ec_png_size;
|
extern const u32 esrb_ec_png_size;
|
||||||
|
|
||||||
extern const u8 esrb_e_png[];
|
extern const u8 esrb_e_png[];
|
||||||
extern const u32 esrb_e_png_size;
|
extern const u32 esrb_e_png_size;
|
||||||
|
|
||||||
extern const u8 esrb_eten_png[];
|
extern const u8 esrb_eten_png[];
|
||||||
extern const u32 esrb_eten_png_size;
|
extern const u32 esrb_eten_png_size;
|
||||||
|
|
||||||
extern const u8 esrb_t_png[];
|
extern const u8 esrb_t_png[];
|
||||||
extern const u32 esrb_t_png_size;
|
extern const u32 esrb_t_png_size;
|
||||||
|
|
||||||
extern const u8 esrb_m_png[];
|
extern const u8 esrb_m_png[];
|
||||||
extern const u32 esrb_m_png_size;
|
extern const u32 esrb_m_png_size;
|
||||||
|
|
||||||
extern const u8 esrb_ao_png[];
|
extern const u8 esrb_ao_png[];
|
||||||
extern const u32 esrb_ao_png_size;
|
extern const u32 esrb_ao_png_size;
|
||||||
|
|
||||||
extern const u8 cero_a_png[];
|
extern const u8 cero_a_png[];
|
||||||
extern const u32 cero_a_png_size;
|
extern const u32 cero_a_png_size;
|
||||||
|
|
||||||
extern const u8 cero_b_png[];
|
extern const u8 cero_b_png[];
|
||||||
extern const u32 cero_b_png_size;
|
extern const u32 cero_b_png_size;
|
||||||
|
|
||||||
extern const u8 cero_c_png[];
|
extern const u8 cero_c_png[];
|
||||||
extern const u32 cero_c_png_size;
|
extern const u32 cero_c_png_size;
|
||||||
|
|
||||||
extern const u8 cero_d_png[];
|
extern const u8 cero_d_png[];
|
||||||
extern const u32 cero_d_png_size;
|
extern const u32 cero_d_png_size;
|
||||||
|
|
||||||
extern const u8 cero_z_png[];
|
extern const u8 cero_z_png[];
|
||||||
extern const u32 cero_z_png_size;
|
extern const u32 cero_z_png_size;
|
||||||
|
|
||||||
extern const u8 pegi_3_png[];
|
extern const u8 pegi_3_png[];
|
||||||
extern const u32 pegi_3_png_size;
|
extern const u32 pegi_3_png_size;
|
||||||
|
|
||||||
extern const u8 pegi_7_png[];
|
extern const u8 pegi_7_png[];
|
||||||
extern const u32 pegi_7_png_size;
|
extern const u32 pegi_7_png_size;
|
||||||
|
|
||||||
extern const u8 pegi_12_png[];
|
extern const u8 pegi_12_png[];
|
||||||
extern const u32 pegi_12_png_size;
|
extern const u32 pegi_12_png_size;
|
||||||
|
|
||||||
extern const u8 pegi_16_png[];
|
extern const u8 pegi_16_png[];
|
||||||
extern const u32 pegi_16_png_size;
|
extern const u32 pegi_16_png_size;
|
||||||
|
|
||||||
extern const u8 pegi_18_png[];
|
extern const u8 pegi_18_png[];
|
||||||
extern const u32 pegi_18_png_size;
|
extern const u32 pegi_18_png_size;
|
||||||
|
|
||||||
extern const u8 usbport_png[];
|
extern const u8 usbport_png[];
|
||||||
extern const u32 usbport_png_size;
|
extern const u32 usbport_png_size;
|
||||||
|
|
||||||
extern const u8 dvd_png[];
|
extern const u8 dvd_png[];
|
||||||
extern const u32 dvd_png_size;
|
extern const u32 dvd_png_size;
|
||||||
|
|
||||||
extern const u8 new_png[];
|
extern const u8 new_png[];
|
||||||
extern const u32 new_png_size;
|
extern const u32 new_png_size;
|
||||||
|
|
||||||
extern const u8 lock_png[];
|
extern const u8 lock_png[];
|
||||||
extern const u32 lock_png_size;
|
extern const u32 lock_png_size;
|
||||||
|
|
||||||
extern const u8 unlock_png[];
|
extern const u8 unlock_png[];
|
||||||
extern const u32 unlock_png_size;
|
extern const u32 unlock_png_size;
|
||||||
|
|
||||||
extern const u8 stub_bin[];
|
extern const u8 stub_bin[];
|
||||||
extern const u32 stub_bin_size;
|
extern const u32 stub_bin_size;
|
||||||
|
|
||||||
extern const u8 fatffs_module_bin[];
|
extern const u8 fatffs_module_bin[];
|
||||||
extern const u32 fatffs_module_bin_size;
|
extern const u32 fatffs_module_bin_size;
|
||||||
|
|
||||||
extern const u8 ehcmodule_frag_v4_bin[];
|
extern const u8 ehcmodule_frag_v4_bin[];
|
||||||
extern const u32 ehcmodule_frag_v4_bin_size;
|
extern const u32 ehcmodule_frag_v4_bin_size;
|
||||||
|
|
||||||
extern const u8 ehcmodule_frag_v5_bin[];
|
extern const u8 ehcmodule_frag_v5_bin[];
|
||||||
extern const u32 ehcmodule_frag_v5_bin_size;
|
extern const u32 ehcmodule_frag_v5_bin_size;
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -10,72 +10,66 @@ bool textVideoInit = false;
|
|||||||
#include <stdarg.h>
|
#include <stdarg.h>
|
||||||
|
|
||||||
//using the gprintf from crediar because it is smaller than mine
|
//using the gprintf from crediar because it is smaller than mine
|
||||||
void gprintf( const char *str, ... )
|
void gprintf(const char *str, ...)
|
||||||
{
|
{
|
||||||
if ( !( geckoinit ) )return;
|
if (!(geckoinit)) return;
|
||||||
|
|
||||||
char astr[ 0x100 ];
|
char astr[0x100];
|
||||||
|
|
||||||
va_list ap;
|
va_list ap;
|
||||||
va_start( ap, str );
|
va_start( ap, str );
|
||||||
|
|
||||||
vsprintf( astr, str, ap );
|
vsprintf(astr, str, ap);
|
||||||
|
|
||||||
va_end( ap );
|
va_end( ap );
|
||||||
|
|
||||||
usb_sendbuffer( 1, astr, strlen( astr ) );
|
usb_sendbuffer(1, astr, strlen(astr));
|
||||||
//usb_sendbuffer_safe( 1, astr, strlen(astr) );
|
//usb_sendbuffer_safe( 1, astr, strlen(astr) );
|
||||||
}
|
}
|
||||||
|
|
||||||
bool InitGecko()
|
bool InitGecko()
|
||||||
{
|
{
|
||||||
u32 geckoattached = usb_isgeckoalive( EXI_CHANNEL_1 );
|
u32 geckoattached = usb_isgeckoalive(EXI_CHANNEL_1);
|
||||||
if ( geckoattached )
|
if (geckoattached)
|
||||||
{
|
{
|
||||||
usb_flush( EXI_CHANNEL_1 );
|
usb_flush(EXI_CHANNEL_1);
|
||||||
CON_EnableGecko( 1, true );
|
CON_EnableGecko(1, true);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
else return false;
|
else return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
char ascii( char s )
|
char ascii(char s)
|
||||||
{
|
{
|
||||||
if ( s < 0x20 )
|
if (s < 0x20) return '.';
|
||||||
return '.';
|
if (s > 0x7E) return '.';
|
||||||
if ( s > 0x7E )
|
|
||||||
return '.';
|
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
void hexdump( void *d, int len )
|
void hexdump(void *d, int len)
|
||||||
{
|
{
|
||||||
u8 *data;
|
u8 *data;
|
||||||
int i, off;
|
int i, off;
|
||||||
data = ( u8* )d;
|
data = (u8*) d;
|
||||||
|
|
||||||
gprintf( "\n 0 1 2 3 4 5 6 7 8 9 A B C D E F 0123456789ABCDEF" );
|
gprintf("\n 0 1 2 3 4 5 6 7 8 9 A B C D E F 0123456789ABCDEF");
|
||||||
gprintf( "\n==== =============================================== ================\n" );
|
gprintf("\n==== =============================================== ================\n");
|
||||||
|
|
||||||
for ( off = 0; off < len; off += 16 )
|
for (off = 0; off < len; off += 16)
|
||||||
{
|
{
|
||||||
gprintf( "%04x ", off );
|
gprintf("%04x ", off);
|
||||||
for ( i = 0; i < 16; i++ )
|
for (i = 0; i < 16; i++)
|
||||||
if ( ( i + off ) >= len )
|
if ((i + off) >= len)
|
||||||
gprintf( " " );
|
gprintf(" ");
|
||||||
else
|
else gprintf("%02x ", data[off + i]);
|
||||||
gprintf( "%02x ", data[off+i] );
|
|
||||||
|
|
||||||
gprintf( " " );
|
gprintf(" ");
|
||||||
for ( i = 0; i < 16; i++ )
|
for (i = 0; i < 16; i++)
|
||||||
if ( ( i + off ) >= len )
|
if ((i + off) >= len)
|
||||||
gprintf( " " );
|
gprintf(" ");
|
||||||
else
|
else gprintf("%c", ascii(data[off + i]));
|
||||||
gprintf( "%c", ascii( data[off+i] ) );
|
gprintf("\n");
|
||||||
gprintf( "\n" );
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#endif /* NO_DEBUG */
|
#endif /* NO_DEBUG */
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
|
|
||||||
|
|
||||||
#ifndef _GECKO_H_
|
#ifndef _GECKO_H_
|
||||||
#define _GECKO_H_
|
#define _GECKO_H_
|
||||||
|
|
||||||
@ -8,21 +7,19 @@ extern "C"
|
|||||||
{
|
{
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
char ascii( char s );
|
char ascii(char s);
|
||||||
|
|
||||||
#ifndef NO_DEBUG
|
#ifndef NO_DEBUG
|
||||||
//use this just like printf();
|
//use this just like printf();
|
||||||
void gprintf( const char *str, ... );
|
void gprintf(const char *str, ...);
|
||||||
bool InitGecko();
|
bool InitGecko();
|
||||||
void hexdump( void *d, int len );
|
void hexdump(void *d, int len);
|
||||||
#else
|
#else
|
||||||
#define gprintf(...)
|
#define gprintf(...)
|
||||||
#define InitGecko() false
|
#define InitGecko() false
|
||||||
#define hexdump( x, y )
|
#define hexdump( x, y )
|
||||||
#endif /* NO_DEBUG */
|
#endif /* NO_DEBUG */
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -44,7 +44,7 @@ void FreeHomebrewBuffer()
|
|||||||
static int SetupARGV( struct __argv * args )
|
static int SetupARGV( struct __argv * args )
|
||||||
{
|
{
|
||||||
if ( !args )
|
if ( !args )
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
bzero( args, sizeof( struct __argv ) );
|
bzero( args, sizeof( struct __argv ) );
|
||||||
args->argvMagic = ARGV_MAGIC;
|
args->argvMagic = ARGV_MAGIC;
|
||||||
@ -60,7 +60,7 @@ static int SetupARGV( struct __argv * args )
|
|||||||
args->length = stringlength;
|
args->length = stringlength;
|
||||||
args->commandLine = ( char* ) malloc( args->length );
|
args->commandLine = ( char* ) malloc( args->length );
|
||||||
if ( !args->commandLine )
|
if ( !args->commandLine )
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
u32 argc = 0;
|
u32 argc = 0;
|
||||||
u32 position = 0;
|
u32 position = 0;
|
||||||
@ -87,7 +87,7 @@ static int SetupARGV( struct __argv * args )
|
|||||||
int BootHomebrew()
|
int BootHomebrew()
|
||||||
{
|
{
|
||||||
if ( homebrewsize <= 0 )
|
if ( homebrewsize <= 0 )
|
||||||
Sys_BackToLoader();
|
Sys_BackToLoader();
|
||||||
|
|
||||||
SDCard_deInit();
|
SDCard_deInit();
|
||||||
USBDevice_deInit();
|
USBDevice_deInit();
|
||||||
@ -101,7 +101,7 @@ int BootHomebrew()
|
|||||||
entrypoint entry = ( entrypoint ) load_dol( app_booter_dol, &args );
|
entrypoint entry = ( entrypoint ) load_dol( app_booter_dol, &args );
|
||||||
|
|
||||||
if ( !entry )
|
if ( !entry )
|
||||||
Sys_BackToLoader();
|
Sys_BackToLoader();
|
||||||
|
|
||||||
VIDEO_SetBlack( true );
|
VIDEO_SetBlack( true );
|
||||||
VIDEO_Flush();
|
VIDEO_Flush();
|
||||||
@ -120,7 +120,7 @@ int BootHomebrew( const char * filepath )
|
|||||||
{
|
{
|
||||||
FILE * file = fopen( filepath, "rb" );
|
FILE * file = fopen( filepath, "rb" );
|
||||||
if ( !file )
|
if ( !file )
|
||||||
Sys_BackToLoader();
|
Sys_BackToLoader();
|
||||||
|
|
||||||
fseek( file, 0, SEEK_END );
|
fseek( file, 0, SEEK_END );
|
||||||
|
|
||||||
@ -153,83 +153,79 @@ int BootHomebrew( const char * filepath )
|
|||||||
#include "fatmounter.h"
|
#include "fatmounter.h"
|
||||||
#include "dolloader.h"
|
#include "dolloader.h"
|
||||||
|
|
||||||
|
|
||||||
void *innetbuffer = NULL;
|
void *innetbuffer = NULL;
|
||||||
static u8 *homebrewbuffer = ( u8 * )0x92000000;
|
static u8 *homebrewbuffer = (u8 *) 0x92000000;
|
||||||
u32 homebrewsize = 0;
|
u32 homebrewsize = 0;
|
||||||
static std::vector<std::string> Arguments;
|
static std::vector<std::string> Arguments;
|
||||||
|
|
||||||
extern const u8 app_booter_dol[];
|
extern const u8 app_booter_dol[];
|
||||||
|
|
||||||
int AllocHomebrewMemory( u32 filesize )
|
int AllocHomebrewMemory(u32 filesize)
|
||||||
{
|
{
|
||||||
|
|
||||||
innetbuffer = malloc( filesize );
|
innetbuffer = malloc(filesize);
|
||||||
|
|
||||||
if ( !innetbuffer )
|
if (!innetbuffer) return -1;
|
||||||
return -1;
|
|
||||||
|
|
||||||
homebrewsize = filesize;
|
homebrewsize = filesize;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
void AddBootArgument( const char * argv )
|
void AddBootArgument(const char * argv)
|
||||||
{
|
{
|
||||||
std::string arg( argv );
|
std::string arg(argv);
|
||||||
Arguments.push_back( arg );
|
Arguments.push_back(arg);
|
||||||
}
|
}
|
||||||
|
|
||||||
int CopyHomebrewMemory( u8 *temp, u32 pos, u32 len )
|
int CopyHomebrewMemory(u8 *temp, u32 pos, u32 len)
|
||||||
{
|
{
|
||||||
homebrewsize += len;
|
homebrewsize += len;
|
||||||
memcpy( ( homebrewbuffer ) + pos, temp, len );
|
memcpy((homebrewbuffer) + pos, temp, len);
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FreeHomebrewBuffer()
|
void FreeHomebrewBuffer()
|
||||||
{
|
{
|
||||||
homebrewbuffer = ( u8 * ) 0x92000000;
|
homebrewbuffer = (u8 *) 0x92000000;
|
||||||
homebrewsize = 0;
|
homebrewsize = 0;
|
||||||
|
|
||||||
if ( innetbuffer )
|
if (innetbuffer)
|
||||||
{
|
{
|
||||||
free( innetbuffer );
|
free(innetbuffer);
|
||||||
innetbuffer = NULL;
|
innetbuffer = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
Arguments.clear();
|
Arguments.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
static int SetupARGV( struct __argv * args )
|
static int SetupARGV(struct __argv * args)
|
||||||
{
|
{
|
||||||
if ( !args )
|
if (!args) return -1;
|
||||||
return -1;
|
|
||||||
|
|
||||||
bzero( args, sizeof( struct __argv ) );
|
bzero(args, sizeof(struct __argv));
|
||||||
args->argvMagic = ARGV_MAGIC;
|
args->argvMagic = ARGV_MAGIC;
|
||||||
|
|
||||||
u32 stringlength = 1;
|
u32 stringlength = 1;
|
||||||
|
|
||||||
/** Append Arguments **/
|
/** Append Arguments **/
|
||||||
for ( u32 i = 0; i < Arguments.size(); i++ )
|
for (u32 i = 0; i < Arguments.size(); i++)
|
||||||
{
|
{
|
||||||
stringlength += Arguments[i].size() + 1;
|
stringlength += Arguments[i].size() + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
args->length = stringlength;
|
args->length = stringlength;
|
||||||
args->commandLine = ( char* ) malloc( args->length );
|
args->commandLine = (char*) malloc(args->length);
|
||||||
|
|
||||||
if ( !args->commandLine )
|
if (!args->commandLine) return -1;
|
||||||
return -1;
|
|
||||||
|
|
||||||
u32 argc = 0;
|
u32 argc = 0;
|
||||||
u32 position = 0;
|
u32 position = 0;
|
||||||
|
|
||||||
/** Append Arguments **/
|
/** Append Arguments **/
|
||||||
for ( u32 i = 0; i < Arguments.size(); i++ )
|
for (u32 i = 0; i < Arguments.size(); i++)
|
||||||
{
|
{
|
||||||
strcpy( &args->commandLine[position], Arguments[i].c_str() );
|
strcpy(&args->commandLine[position], Arguments[i].c_str());
|
||||||
position += Arguments[i].size() + 1;
|
position += Arguments[i].size() + 1;
|
||||||
argc++;
|
argc++;
|
||||||
}
|
}
|
||||||
@ -247,17 +243,16 @@ static int SetupARGV( struct __argv * args )
|
|||||||
|
|
||||||
static int RunAppbooter()
|
static int RunAppbooter()
|
||||||
{
|
{
|
||||||
if ( homebrewsize == 0 )
|
if (homebrewsize == 0) return -1;
|
||||||
return -1;
|
|
||||||
|
|
||||||
struct __argv args;
|
struct __argv args;
|
||||||
SetupARGV( &args );
|
SetupARGV(&args);
|
||||||
|
|
||||||
u32 cpu_isr;
|
u32 cpu_isr;
|
||||||
|
|
||||||
entrypoint entry = ( entrypoint ) load_dol( ( void* ) app_booter_dol, &args );
|
entrypoint entry = (entrypoint) load_dol((void*) app_booter_dol, &args);
|
||||||
|
|
||||||
if ( !entry )
|
if (!entry)
|
||||||
{
|
{
|
||||||
FreeHomebrewBuffer();
|
FreeHomebrewBuffer();
|
||||||
return -1;
|
return -1;
|
||||||
@ -266,25 +261,24 @@ static int RunAppbooter()
|
|||||||
u64 currentStub = getStubDest();
|
u64 currentStub = getStubDest();
|
||||||
loadStub();
|
loadStub();
|
||||||
|
|
||||||
if ( Set_Stub_Split( 0x00010001, "UNEO" ) < 0 )
|
if (Set_Stub_Split(0x00010001, "UNEO") < 0)
|
||||||
{
|
{
|
||||||
if ( Set_Stub_Split( 0x00010001, "ULNR" ) < 0 )
|
if (Set_Stub_Split(0x00010001, "ULNR") < 0)
|
||||||
{
|
{
|
||||||
if ( !currentStub )
|
if (!currentStub) currentStub = 0x100000002ULL;
|
||||||
currentStub = 0x100000002ULL;
|
|
||||||
|
|
||||||
Set_Stub( currentStub );
|
Set_Stub(currentStub);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
SDCard_deInit();
|
SDCard_deInit();
|
||||||
USBDevice_deInit();
|
USBDevice_deInit();
|
||||||
|
|
||||||
WPAD_Flush( 0 );
|
WPAD_Flush(0);
|
||||||
WPAD_Disconnect( 0 );
|
WPAD_Disconnect(0);
|
||||||
WPAD_Shutdown();
|
WPAD_Shutdown();
|
||||||
|
|
||||||
SYS_ResetSystem( SYS_SHUTDOWN, 0, 0 );
|
SYS_ResetSystem(SYS_SHUTDOWN, 0, 0);
|
||||||
_CPU_ISR_Disable( cpu_isr );
|
_CPU_ISR_Disable( cpu_isr );
|
||||||
__exception_closeall();
|
__exception_closeall();
|
||||||
entry();
|
entry();
|
||||||
@ -293,59 +287,58 @@ static int RunAppbooter()
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int BootHomebrew( char * filepath )
|
int BootHomebrew(char * filepath)
|
||||||
{
|
{
|
||||||
void *buffer = NULL;
|
void *buffer = NULL;
|
||||||
u32 filesize = 0;
|
u32 filesize = 0;
|
||||||
|
|
||||||
FILE *file = fopen( filepath, "rb" );
|
FILE *file = fopen(filepath, "rb");
|
||||||
|
|
||||||
if ( !file )
|
if (!file) Sys_BackToLoader();
|
||||||
Sys_BackToLoader();
|
|
||||||
|
|
||||||
fseek( file, 0, SEEK_END );
|
fseek(file, 0, SEEK_END);
|
||||||
filesize = ftell( file );
|
filesize = ftell(file);
|
||||||
rewind( file );
|
rewind(file);
|
||||||
|
|
||||||
buffer = malloc( filesize );
|
buffer = malloc(filesize);
|
||||||
|
|
||||||
if ( fread( buffer, 1, filesize, file ) != filesize )
|
if (fread(buffer, 1, filesize, file) != filesize)
|
||||||
{
|
{
|
||||||
fclose( file );
|
fclose(file);
|
||||||
free( buffer );
|
free(buffer);
|
||||||
SDCard_deInit();
|
SDCard_deInit();
|
||||||
USBDevice_deInit();
|
USBDevice_deInit();
|
||||||
Sys_BackToLoader();
|
Sys_BackToLoader();
|
||||||
}
|
}
|
||||||
|
|
||||||
fclose( file );
|
fclose(file);
|
||||||
|
|
||||||
CopyHomebrewMemory( ( u8* ) buffer, 0, filesize );
|
CopyHomebrewMemory((u8*) buffer, 0, filesize);
|
||||||
|
|
||||||
if ( buffer )
|
if (buffer)
|
||||||
{
|
{
|
||||||
free( buffer );
|
free(buffer);
|
||||||
buffer = NULL;
|
buffer = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
AddBootArgument( filepath );
|
AddBootArgument(filepath);
|
||||||
return RunAppbooter();
|
return RunAppbooter();
|
||||||
}
|
}
|
||||||
|
|
||||||
int BootHomebrewFromMem()
|
int BootHomebrewFromMem()
|
||||||
{
|
{
|
||||||
gprintf( "BootHomebrewFromMem()\n %p, %08x\n", innetbuffer, homebrewsize );
|
gprintf("BootHomebrewFromMem()\n %p, %08x\n", innetbuffer, homebrewsize);
|
||||||
|
|
||||||
if ( !innetbuffer )
|
if (!innetbuffer)
|
||||||
{
|
{
|
||||||
gprintf( "!innetbuffer\n" );
|
gprintf("!innetbuffer\n");
|
||||||
SDCard_deInit();
|
SDCard_deInit();
|
||||||
USBDevice_deInit();
|
USBDevice_deInit();
|
||||||
Sys_BackToLoader();
|
Sys_BackToLoader();
|
||||||
}
|
}
|
||||||
|
|
||||||
CopyHomebrewMemory( ( u8* ) innetbuffer, 0, homebrewsize );
|
CopyHomebrewMemory((u8*) innetbuffer, 0, homebrewsize);
|
||||||
free( innetbuffer );
|
free(innetbuffer);
|
||||||
|
|
||||||
return RunAppbooter();
|
return RunAppbooter();
|
||||||
}
|
}
|
||||||
|
@ -3,12 +3,12 @@
|
|||||||
|
|
||||||
//int BootHomebrew();
|
//int BootHomebrew();
|
||||||
int BootHomebrewFromMem();
|
int BootHomebrewFromMem();
|
||||||
int BootHomebrew( char * filepath );
|
int BootHomebrew(char * filepath);
|
||||||
int CopyHomebrewMemory( u8 *temp, u32 pos, u32 len );
|
int CopyHomebrewMemory(u8 *temp, u32 pos, u32 len);
|
||||||
void AddBootArgument( const char * arg );
|
void AddBootArgument(const char * arg);
|
||||||
void FreeHomebrewBuffer();
|
void FreeHomebrewBuffer();
|
||||||
int LoadHomebrew( const char * filepath );
|
int LoadHomebrew(const char * filepath);
|
||||||
int AllocHomebrewMemory( u32 filesize );
|
int AllocHomebrewMemory(u32 filesize);
|
||||||
extern void *innetbuffer;
|
extern void *innetbuffer;
|
||||||
extern u32 homebrewsize;
|
extern u32 homebrewsize;
|
||||||
|
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -10,7 +10,7 @@
|
|||||||
#ifndef _HOMEBREWBROWSE_H_
|
#ifndef _HOMEBREWBROWSE_H_
|
||||||
#define _HOMEBREWBROWSE_H_
|
#define _HOMEBREWBROWSE_H_
|
||||||
|
|
||||||
int roundup( float number );
|
int roundup(float number);
|
||||||
int MenuHomebrewBrowse();
|
int MenuHomebrewBrowse();
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -9,113 +9,110 @@
|
|||||||
|
|
||||||
#include "HomebrewFiles.h"
|
#include "HomebrewFiles.h"
|
||||||
|
|
||||||
HomebrewFiles::HomebrewFiles( const char * path )
|
HomebrewFiles::HomebrewFiles(const char * path)
|
||||||
{
|
{
|
||||||
filecount = 0;
|
filecount = 0;
|
||||||
|
|
||||||
FileInfo = ( FileInfos * ) malloc( sizeof( FileInfos ) );
|
FileInfo = (FileInfos *) malloc(sizeof(FileInfos));
|
||||||
if ( !FileInfo )
|
if (!FileInfo)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
memset( &FileInfo[filecount], 0, sizeof( FileInfos ) );
|
memset(&FileInfo[filecount], 0, sizeof(FileInfos));
|
||||||
|
|
||||||
this->LoadPath( path );
|
this->LoadPath(path);
|
||||||
this->SortList();
|
this->SortList();
|
||||||
}
|
}
|
||||||
|
|
||||||
HomebrewFiles::~HomebrewFiles()
|
HomebrewFiles::~HomebrewFiles()
|
||||||
{
|
{
|
||||||
if ( FileInfo )
|
if (FileInfo)
|
||||||
{
|
{
|
||||||
free( FileInfo );
|
free(FileInfo);
|
||||||
FileInfo = NULL;
|
FileInfo = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool HomebrewFiles::LoadPath( const char * folderpath )
|
bool HomebrewFiles::LoadPath(const char * folderpath)
|
||||||
{
|
{
|
||||||
struct stat st;
|
struct stat st;
|
||||||
DIR_ITER *dir = NULL;
|
DIR_ITER *dir = NULL;
|
||||||
char filename[1024];
|
char filename[1024];
|
||||||
|
|
||||||
dir = diropen( folderpath );
|
dir = diropen(folderpath);
|
||||||
if ( dir == NULL )
|
if (dir == NULL)
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
while ( dirnext( dir, filename, &st ) == 0 )
|
while (dirnext(dir, filename, &st) == 0)
|
||||||
{
|
{
|
||||||
if ( ( st.st_mode & S_IFDIR ) != 0 )
|
if ((st.st_mode & S_IFDIR) != 0)
|
||||||
{
|
{
|
||||||
if ( strcmp( filename, "." ) != 0 && strcmp( filename, ".." ) != 0 )
|
if (strcmp(filename, ".") != 0 && strcmp(filename, "..") != 0)
|
||||||
{
|
{
|
||||||
char currentname[200];
|
char currentname[200];
|
||||||
snprintf( currentname, sizeof( currentname ), "%s%s/", folderpath, filename );
|
snprintf(currentname, sizeof(currentname), "%s%s/", folderpath, filename);
|
||||||
this->LoadPath( currentname );
|
this->LoadPath(currentname);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
char temp[5];
|
char temp[5];
|
||||||
for ( int i = 0; i < 5; i++ )
|
for (int i = 0; i < 5; i++)
|
||||||
{
|
{
|
||||||
temp[i] = filename[strlen( filename )-4+i];
|
temp[i] = filename[strlen(filename) - 4 + i];
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( ( strncasecmp( temp, ".dol", 4 ) == 0 || strncasecmp( temp, ".elf", 4 ) == 0 )
|
if ((strncasecmp(temp, ".dol", 4) == 0 || strncasecmp(temp, ".elf", 4) == 0) && filecount < MAXHOMEBREWS
|
||||||
&& filecount < MAXHOMEBREWS && filename[0] != '.' )
|
&& filename[0] != '.')
|
||||||
{
|
{
|
||||||
|
|
||||||
FileInfo = ( FileInfos * ) realloc( FileInfo, ( filecount + 1 ) * sizeof( FileInfos ) );
|
FileInfo = (FileInfos *) realloc(FileInfo, (filecount + 1) * sizeof(FileInfos));
|
||||||
|
|
||||||
if ( !FileInfo )
|
if (!FileInfo)
|
||||||
{
|
{
|
||||||
free( FileInfo );
|
free(FileInfo);
|
||||||
FileInfo = NULL;
|
FileInfo = NULL;
|
||||||
filecount = 0;
|
filecount = 0;
|
||||||
dirclose( dir );
|
dirclose(dir);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
memset( &( FileInfo[filecount] ), 0, sizeof( FileInfo ) );
|
memset(&(FileInfo[filecount]), 0, sizeof(FileInfo));
|
||||||
|
|
||||||
strlcpy( FileInfo[filecount].FilePath, folderpath, sizeof( FileInfo[filecount].FilePath ) );
|
strlcpy(FileInfo[filecount].FilePath, folderpath, sizeof(FileInfo[filecount].FilePath));
|
||||||
strlcpy( FileInfo[filecount].FileName, filename, sizeof( FileInfo[filecount].FileName ) );
|
strlcpy(FileInfo[filecount].FileName, filename, sizeof(FileInfo[filecount].FileName));
|
||||||
FileInfo[filecount].FileSize = st.st_size;
|
FileInfo[filecount].FileSize = st.st_size;
|
||||||
filecount++;
|
filecount++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
dirclose( dir );
|
dirclose(dir);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
char * HomebrewFiles::GetFilename( int ind )
|
char * HomebrewFiles::GetFilename(int ind)
|
||||||
{
|
{
|
||||||
if ( ind > filecount )
|
if (ind > filecount)
|
||||||
return NULL;
|
return NULL;
|
||||||
else
|
else return FileInfo[ind].FileName;
|
||||||
return FileInfo[ind].FileName;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
char * HomebrewFiles::GetFilepath( int ind )
|
char * HomebrewFiles::GetFilepath(int ind)
|
||||||
{
|
{
|
||||||
if ( ind > filecount )
|
if (ind > filecount)
|
||||||
return NULL;
|
return NULL;
|
||||||
else
|
else return FileInfo[ind].FilePath;
|
||||||
return FileInfo[ind].FilePath;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int HomebrewFiles::GetFilesize( int ind )
|
unsigned int HomebrewFiles::GetFilesize(int ind)
|
||||||
{
|
{
|
||||||
if ( ind > filecount || !filecount || !FileInfo )
|
if (ind > filecount || !filecount || !FileInfo)
|
||||||
return NULL;
|
return NULL;
|
||||||
else
|
else return FileInfo[ind].FileSize;
|
||||||
return FileInfo[ind].FileSize;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int HomebrewFiles::GetFilecount()
|
int HomebrewFiles::GetFilecount()
|
||||||
@ -123,14 +120,14 @@ int HomebrewFiles::GetFilecount()
|
|||||||
return filecount;
|
return filecount;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ListCompare( const void *a, const void *b )
|
static int ListCompare(const void *a, const void *b)
|
||||||
{
|
{
|
||||||
FileInfos *ab = ( FileInfos* ) a;
|
FileInfos *ab = (FileInfos*) a;
|
||||||
FileInfos *bb = ( FileInfos* ) b;
|
FileInfos *bb = (FileInfos*) b;
|
||||||
|
|
||||||
return stricmp( ( char * ) ab->FilePath, ( char * ) bb->FilePath );
|
return stricmp((char *) ab->FilePath, (char *) bb->FilePath);
|
||||||
}
|
}
|
||||||
void HomebrewFiles::SortList()
|
void HomebrewFiles::SortList()
|
||||||
{
|
{
|
||||||
qsort( FileInfo, filecount, sizeof( FileInfos ), ListCompare );
|
qsort(FileInfo, filecount, sizeof(FileInfos), ListCompare);
|
||||||
}
|
}
|
||||||
|
@ -9,9 +9,9 @@
|
|||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
char FileName[100];
|
char FileName[100];
|
||||||
char FilePath[150];
|
char FilePath[150];
|
||||||
unsigned int FileSize;
|
unsigned int FileSize;
|
||||||
} FileInfos;
|
} FileInfos;
|
||||||
|
|
||||||
class HomebrewFiles
|
class HomebrewFiles
|
||||||
@ -19,21 +19,21 @@ class HomebrewFiles
|
|||||||
public:
|
public:
|
||||||
//!Constructor
|
//!Constructor
|
||||||
//!\param path Path where to check for homebrew files
|
//!\param path Path where to check for homebrew files
|
||||||
HomebrewFiles( const char * path );
|
HomebrewFiles(const char * path);
|
||||||
//!Destructor
|
//!Destructor
|
||||||
~HomebrewFiles();
|
~HomebrewFiles();
|
||||||
//! Load the dol/elf list of a path
|
//! Load the dol/elf list of a path
|
||||||
//!\param path Path where to check for homebrew files
|
//!\param path Path where to check for homebrew files
|
||||||
bool LoadPath( const char * path );
|
bool LoadPath(const char * path);
|
||||||
//! Get the a filename of the list
|
//! Get the a filename of the list
|
||||||
//!\param list index
|
//!\param list index
|
||||||
char * GetFilename( int index );
|
char * GetFilename(int index);
|
||||||
//! Get the a filepath of the list
|
//! Get the a filepath of the list
|
||||||
//!\param list index
|
//!\param list index
|
||||||
char * GetFilepath( int index );
|
char * GetFilepath(int index);
|
||||||
//! Get the a filesize of the list
|
//! Get the a filesize of the list
|
||||||
//!\param list index
|
//!\param list index
|
||||||
unsigned int GetFilesize( int index );
|
unsigned int GetFilesize(int index);
|
||||||
//! Get the filecount of the whole list
|
//! Get the filecount of the whole list
|
||||||
int GetFilecount();
|
int GetFilecount();
|
||||||
//! Sort list by filepath
|
//! Sort list by filepath
|
||||||
|
@ -11,60 +11,110 @@
|
|||||||
|
|
||||||
#define ENTRIE_SIZE 8192
|
#define ENTRIE_SIZE 8192
|
||||||
|
|
||||||
int HomebrewXML::LoadHomebrewXMLData( const char* filename )
|
/* Initializes a new instance of the HomebrewXML class. */
|
||||||
|
HomebrewXML::HomebrewXML()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Finalizes an instance of the HomebrewXML class. */
|
||||||
|
HomebrewXML::~HomebrewXML()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/* qparam filename Filepath of the XML file */
|
||||||
|
int HomebrewXML::LoadHomebrewXMLData(const char* filename)
|
||||||
{
|
{
|
||||||
mxml_node_t *nodedataHB = NULL;
|
mxml_node_t *nodedataHB = NULL;
|
||||||
mxml_node_t *nodetreeHB = NULL;
|
mxml_node_t *nodetreeHB = NULL;
|
||||||
|
|
||||||
/* Load XML file */
|
/* Load XML file */
|
||||||
FILE *filexml;
|
FILE *filexml;
|
||||||
filexml = fopen( filename, "rb" );
|
filexml = fopen(filename, "rb");
|
||||||
if ( !filexml )
|
if (!filexml) return -1;
|
||||||
return -1;
|
|
||||||
|
|
||||||
nodetreeHB = mxmlLoadFile( NULL, filexml, MXML_OPAQUE_CALLBACK );
|
nodetreeHB = mxmlLoadFile(NULL, filexml, MXML_OPAQUE_CALLBACK);
|
||||||
fclose( filexml );
|
fclose(filexml);
|
||||||
|
|
||||||
if ( nodetreeHB == NULL )
|
if (nodetreeHB == NULL) return -2;
|
||||||
return -2;
|
|
||||||
|
|
||||||
nodedataHB = mxmlFindElement( nodetreeHB, nodetreeHB, "app", NULL, NULL, MXML_DESCEND );
|
nodedataHB = mxmlFindElement(nodetreeHB, nodetreeHB, "app", NULL, NULL, MXML_DESCEND);
|
||||||
if ( nodedataHB == NULL )
|
if (nodedataHB == NULL) return -5;
|
||||||
return -5;
|
|
||||||
|
|
||||||
char * Entrie = new char[ENTRIE_SIZE];
|
char * Entrie = new char[ENTRIE_SIZE];
|
||||||
|
|
||||||
GetTextFromNode( nodedataHB, nodedataHB, ( char* ) "name", NULL, NULL, MXML_DESCEND, Entrie, ENTRIE_SIZE );
|
GetTextFromNode(nodedataHB, nodedataHB, (char*) "name", NULL, NULL, MXML_DESCEND, Entrie, ENTRIE_SIZE);
|
||||||
Name = Entrie;
|
Name = Entrie;
|
||||||
|
|
||||||
GetTextFromNode( nodedataHB, nodedataHB, ( char* ) "coder", NULL, NULL, MXML_DESCEND, Entrie, ENTRIE_SIZE );
|
GetTextFromNode(nodedataHB, nodedataHB, (char*) "coder", NULL, NULL, MXML_DESCEND, Entrie, ENTRIE_SIZE);
|
||||||
Coder = Entrie;
|
Coder = Entrie;
|
||||||
|
|
||||||
GetTextFromNode( nodedataHB, nodedataHB, ( char* ) "version", NULL, NULL, MXML_DESCEND, Entrie, ENTRIE_SIZE );
|
GetTextFromNode(nodedataHB, nodedataHB, (char*) "version", NULL, NULL, MXML_DESCEND, Entrie, ENTRIE_SIZE);
|
||||||
Version = Entrie;
|
Version = Entrie;
|
||||||
|
|
||||||
GetTextFromNode( nodedataHB, nodedataHB, ( char* ) "short_description", NULL, NULL, MXML_DESCEND, Entrie, ENTRIE_SIZE );
|
GetTextFromNode(nodedataHB, nodedataHB, (char*) "short_description", NULL, NULL, MXML_DESCEND, Entrie, ENTRIE_SIZE);
|
||||||
ShortDescription = Entrie;
|
ShortDescription = Entrie;
|
||||||
|
|
||||||
GetTextFromNode( nodedataHB, nodedataHB, ( char* ) "long_description", NULL, NULL, MXML_DESCEND, Entrie, ENTRIE_SIZE );
|
GetTextFromNode(nodedataHB, nodedataHB, (char*) "long_description", NULL, NULL, MXML_DESCEND, Entrie, ENTRIE_SIZE);
|
||||||
LongDescription = Entrie;
|
LongDescription = Entrie;
|
||||||
|
|
||||||
GetTextFromNode( nodedataHB, nodedataHB, ( char* ) "release_date", NULL, NULL, MXML_DESCEND, Entrie, ENTRIE_SIZE );
|
GetTextFromNode(nodedataHB, nodedataHB, (char*) "release_date", NULL, NULL, MXML_DESCEND, Entrie, ENTRIE_SIZE);
|
||||||
|
|
||||||
int len = ( strlen( Entrie ) - 6 ); //length of the date string without the 200000 at the end
|
int len = (strlen(Entrie) - 6); //length of the date string without the 200000 at the end
|
||||||
if ( len == 8 )
|
if (len == 8)
|
||||||
snprintf( Entrie, ENTRIE_SIZE, "%c%c/%c%c/%c%c%c%c", Entrie[4], Entrie[5], Entrie[6], Entrie[7], Entrie[0], Entrie[1], Entrie[2], Entrie[3] );
|
snprintf(Entrie, ENTRIE_SIZE, "%c%c/%c%c/%c%c%c%c", Entrie[4], Entrie[5], Entrie[6], Entrie[7], Entrie[0],
|
||||||
else if ( len == 6 )
|
Entrie[1], Entrie[2], Entrie[3]);
|
||||||
snprintf( Entrie, ENTRIE_SIZE, "%c%c/%c%c%c%c", Entrie[4], Entrie[5], Entrie[0], Entrie[1], Entrie[2], Entrie[3] );
|
else if (len == 6)
|
||||||
else
|
snprintf(Entrie, ENTRIE_SIZE, "%c%c/%c%c%c%c", Entrie[4], Entrie[5], Entrie[0], Entrie[1], Entrie[2], Entrie[3]);
|
||||||
snprintf( Entrie, ENTRIE_SIZE, "%s", Entrie );
|
else snprintf(Entrie, ENTRIE_SIZE, "%s", Entrie);
|
||||||
|
|
||||||
Releasedate = Entrie;
|
Releasedate = Entrie;
|
||||||
|
|
||||||
free( nodedataHB );
|
free(nodedataHB);
|
||||||
free( nodetreeHB );
|
free(nodetreeHB);
|
||||||
|
|
||||||
delete [] Entrie;
|
delete[] Entrie;
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Get name */
|
||||||
|
const char * HomebrewXML::GetName()
|
||||||
|
{
|
||||||
|
return Name.c_str();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set Name */
|
||||||
|
void HomebrewXML::SetName(char * newName)
|
||||||
|
{
|
||||||
|
Name = newName;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get coder */
|
||||||
|
const char * HomebrewXML::GetCoder()
|
||||||
|
{
|
||||||
|
return Coder.c_str();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get version */
|
||||||
|
const char * HomebrewXML::GetVersion()
|
||||||
|
{
|
||||||
|
return Version.c_str();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get releasedate */
|
||||||
|
const char * HomebrewXML::GetReleasedate()
|
||||||
|
{
|
||||||
|
return Releasedate.c_str();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get shortdescription */
|
||||||
|
const char * HomebrewXML::GetShortDescription()
|
||||||
|
{
|
||||||
|
return ShortDescription.c_str();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get longdescription */
|
||||||
|
const char * HomebrewXML::GetLongDescription()
|
||||||
|
{
|
||||||
|
return LongDescription.c_str();
|
||||||
|
}
|
||||||
|
@ -10,27 +10,19 @@
|
|||||||
class HomebrewXML
|
class HomebrewXML
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
//!Constructor
|
HomebrewXML();
|
||||||
//!\param path Path for the xml file
|
~HomebrewXML();
|
||||||
HomebrewXML() { };
|
|
||||||
//!Destructor
|
int LoadHomebrewXMLData(const char* filename);
|
||||||
~HomebrewXML() { };
|
|
||||||
//!\param filename Filepath of the XML file
|
const char * GetName();
|
||||||
int LoadHomebrewXMLData( const char* filename );
|
void SetName(char * newName);
|
||||||
//! Get name
|
const char * GetCoder();
|
||||||
const char * GetName() { return Name.c_str(); };
|
const char * GetVersion();
|
||||||
//! Get coder
|
const char * GetReleasedate();
|
||||||
const char * GetCoder() { return Coder.c_str(); };
|
const char * GetShortDescription();
|
||||||
//! Get version
|
const char * GetLongDescription();
|
||||||
const char * GetVersion() { return Version.c_str(); };
|
|
||||||
//! Get releasedate
|
|
||||||
const char * GetReleasedate() { return Releasedate.c_str(); };
|
|
||||||
//! Get shortdescription
|
|
||||||
const char * GetShortDescription() { return ShortDescription.c_str(); };
|
|
||||||
//! Get longdescription
|
|
||||||
const char * GetLongDescription() { return LongDescription.c_str(); };
|
|
||||||
//! Set Name
|
|
||||||
void SetName( char * newName ) { Name = newName; };
|
|
||||||
protected:
|
protected:
|
||||||
std::string Name;
|
std::string Name;
|
||||||
std::string Coder;
|
std::string Coder;
|
||||||
|
@ -21,36 +21,36 @@ typedef struct _dolheader
|
|||||||
u32 entry_point;
|
u32 entry_point;
|
||||||
} dolheader;
|
} dolheader;
|
||||||
|
|
||||||
u32 load_dol( const void *dolstart, struct __argv *argv )
|
u32 load_dol(const void *dolstart, struct __argv *argv)
|
||||||
{
|
{
|
||||||
u32 i;
|
u32 i;
|
||||||
dolheader *dolfile;
|
dolheader *dolfile;
|
||||||
|
|
||||||
if ( dolstart )
|
if (dolstart)
|
||||||
{
|
{
|
||||||
dolfile = ( dolheader * ) dolstart;
|
dolfile = (dolheader *) dolstart;
|
||||||
for ( i = 0; i < 7; i++ )
|
for (i = 0; i < 7; i++)
|
||||||
{
|
{
|
||||||
if ( ( !dolfile->text_size[i] ) || ( dolfile->text_start[i] < 0x100 ) ) continue;
|
if ((!dolfile->text_size[i]) || (dolfile->text_start[i] < 0x100)) continue;
|
||||||
ICInvalidateRange ( ( void * ) dolfile->text_start[i], dolfile->text_size[i] );
|
ICInvalidateRange((void *) dolfile->text_start[i], dolfile->text_size[i]);
|
||||||
memcpy( ( void * ) dolfile->text_start[i], dolstart + dolfile->text_pos[i], dolfile->text_size[i] );
|
memcpy((void *) dolfile->text_start[i], dolstart + dolfile->text_pos[i], dolfile->text_size[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
for ( i = 0; i < 11; i++ )
|
for (i = 0; i < 11; i++)
|
||||||
{
|
{
|
||||||
if ( ( !dolfile->data_size[i] ) || ( dolfile->data_start[i] < 0x100 ) ) continue;
|
if ((!dolfile->data_size[i]) || (dolfile->data_start[i] < 0x100)) continue;
|
||||||
memcpy( ( void * ) dolfile->data_start[i], dolstart + dolfile->data_pos[i], dolfile->data_size[i] );
|
memcpy((void *) dolfile->data_start[i], dolstart + dolfile->data_pos[i], dolfile->data_size[i]);
|
||||||
DCFlushRangeNoSync ( ( void * ) dolfile->data_start[i], dolfile->data_size[i] );
|
DCFlushRangeNoSync((void *) dolfile->data_start[i], dolfile->data_size[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
memset ( ( void * ) dolfile->bss_start, 0, dolfile->bss_size );
|
memset((void *) dolfile->bss_start, 0, dolfile->bss_size);
|
||||||
DCFlushRange( ( void * ) dolfile->bss_start, dolfile->bss_size );
|
DCFlushRange((void *) dolfile->bss_start, dolfile->bss_size);
|
||||||
|
|
||||||
if ( argv && argv->argvMagic == ARGV_MAGIC )
|
if (argv && argv->argvMagic == ARGV_MAGIC)
|
||||||
{
|
{
|
||||||
void *new_argv = ( void * )( dolfile->entry_point + 8 );
|
void *new_argv = (void *) (dolfile->entry_point + 8);
|
||||||
memcpy( new_argv, argv, sizeof( *argv ) );
|
memcpy(new_argv, argv, sizeof(*argv));
|
||||||
DCFlushRange( new_argv, sizeof( *argv ) );
|
DCFlushRange(new_argv, sizeof(*argv));
|
||||||
}
|
}
|
||||||
|
|
||||||
return dolfile->entry_point;
|
return dolfile->entry_point;
|
||||||
|
@ -7,10 +7,9 @@ extern "C"
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
extern void __exception_closeall();
|
extern void __exception_closeall();
|
||||||
typedef void ( *entrypoint ) ( void );
|
typedef void (*entrypoint)(void);
|
||||||
|
|
||||||
u32 load_dol( const void *dolstart, struct __argv *argv );
|
|
||||||
|
|
||||||
|
u32 load_dol(const void *dolstart, struct __argv *argv);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
|
@ -20,9 +20,9 @@
|
|||||||
#include "input.h"
|
#include "input.h"
|
||||||
#include "libwiigui/gui.h"
|
#include "libwiigui/gui.h"
|
||||||
|
|
||||||
int rumbleRequest[4] = {0, 0, 0, 0};
|
int rumbleRequest[4] = { 0, 0, 0, 0 };
|
||||||
GuiTrigger userInput[4];
|
GuiTrigger userInput[4];
|
||||||
static int rumbleCount[4] = {0, 0, 0, 0};
|
static int rumbleCount[4] = { 0, 0, 0, 0 };
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
* UpdatePads
|
* UpdatePads
|
||||||
@ -34,22 +34,21 @@ void UpdatePads()
|
|||||||
WPAD_ScanPads();
|
WPAD_ScanPads();
|
||||||
PAD_ScanPads();
|
PAD_ScanPads();
|
||||||
|
|
||||||
for ( int i = 3; i >= 0; i-- )
|
for (int i = 3; i >= 0; i--)
|
||||||
{
|
{
|
||||||
memcpy( &userInput[i].wpad, WPAD_Data( i ), sizeof( WPADData ) );
|
memcpy(&userInput[i].wpad, WPAD_Data(i), sizeof(WPADData));
|
||||||
userInput[i].chan = i;
|
userInput[i].chan = i;
|
||||||
userInput[i].pad.btns_d = PAD_ButtonsDown( i );
|
userInput[i].pad.btns_d = PAD_ButtonsDown(i);
|
||||||
userInput[i].pad.btns_u = PAD_ButtonsUp( i );
|
userInput[i].pad.btns_u = PAD_ButtonsUp(i);
|
||||||
userInput[i].pad.btns_h = PAD_ButtonsHeld( i );
|
userInput[i].pad.btns_h = PAD_ButtonsHeld(i);
|
||||||
userInput[i].pad.stickX = PAD_StickX( i );
|
userInput[i].pad.stickX = PAD_StickX(i);
|
||||||
userInput[i].pad.stickY = PAD_StickY( i );
|
userInput[i].pad.stickY = PAD_StickY(i);
|
||||||
userInput[i].pad.substickX = PAD_SubStickX( i );
|
userInput[i].pad.substickX = PAD_SubStickX(i);
|
||||||
userInput[i].pad.substickY = PAD_SubStickY( i );
|
userInput[i].pad.substickY = PAD_SubStickY(i);
|
||||||
userInput[i].pad.triggerL = PAD_TriggerL( i );
|
userInput[i].pad.triggerL = PAD_TriggerL(i);
|
||||||
userInput[i].pad.triggerR = PAD_TriggerR( i );
|
userInput[i].pad.triggerR = PAD_TriggerR(i);
|
||||||
|
|
||||||
if ( Settings.rumble == RumbleOn )
|
if (Settings.rumble == RumbleOn) DoRumble(i);
|
||||||
DoRumble( i );
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -64,10 +63,10 @@ void SetupPads()
|
|||||||
WPAD_Init();
|
WPAD_Init();
|
||||||
|
|
||||||
// read wiimote accelerometer and IR data
|
// read wiimote accelerometer and IR data
|
||||||
WPAD_SetDataFormat( WPAD_CHAN_ALL, WPAD_FMT_BTNS_ACC_IR );
|
WPAD_SetDataFormat(WPAD_CHAN_ALL, WPAD_FMT_BTNS_ACC_IR);
|
||||||
WPAD_SetVRes( WPAD_CHAN_ALL, screenwidth, screenheight );
|
WPAD_SetVRes(WPAD_CHAN_ALL, screenwidth, screenheight);
|
||||||
|
|
||||||
for ( int i = 0; i < 4; i++ )
|
for (int i = 0; i < 4; i++)
|
||||||
{
|
{
|
||||||
userInput[i].chan = i;
|
userInput[i].chan = i;
|
||||||
}
|
}
|
||||||
@ -79,9 +78,9 @@ void SetupPads()
|
|||||||
|
|
||||||
void ShutoffRumble()
|
void ShutoffRumble()
|
||||||
{
|
{
|
||||||
for ( int i = 0; i < 4; i++ )
|
for (int i = 0; i < 4; i++)
|
||||||
{
|
{
|
||||||
WPAD_Rumble( i, 0 );
|
WPAD_Rumble(i, 0);
|
||||||
rumbleCount[i] = 0;
|
rumbleCount[i] = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -90,23 +89,22 @@ void ShutoffRumble()
|
|||||||
* DoRumble
|
* DoRumble
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
void DoRumble( int i )
|
void DoRumble(int i)
|
||||||
{
|
{
|
||||||
if ( rumbleRequest[i] && rumbleCount[i] < 3 )
|
if (rumbleRequest[i] && rumbleCount[i] < 3)
|
||||||
{
|
{
|
||||||
WPAD_Rumble( i, 1 ); // rumble on
|
WPAD_Rumble(i, 1); // rumble on
|
||||||
rumbleCount[i]++;
|
rumbleCount[i]++;
|
||||||
}
|
}
|
||||||
else if ( rumbleRequest[i] )
|
else if (rumbleRequest[i])
|
||||||
{
|
{
|
||||||
rumbleCount[i] = 20;
|
rumbleCount[i] = 20;
|
||||||
rumbleRequest[i] = 0;
|
rumbleRequest[i] = 0;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if ( rumbleCount[i] )
|
if (rumbleCount[i]) rumbleCount[i]--;
|
||||||
rumbleCount[i]--;
|
WPAD_Rumble(i, 0); // rumble off
|
||||||
WPAD_Rumble( i, 0 ); // rumble off
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -116,17 +114,17 @@ void DoRumble( int i )
|
|||||||
* Get X/Y value from Wii Joystick (classic, nunchuk) input
|
* Get X/Y value from Wii Joystick (classic, nunchuk) input
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
s8 WPAD_Stick( u8 chan, u8 right, int axis )
|
s8 WPAD_Stick(u8 chan, u8 right, int axis)
|
||||||
{
|
{
|
||||||
float mag = 0.0;
|
float mag = 0.0;
|
||||||
float ang = 0.0;
|
float ang = 0.0;
|
||||||
WPADData *data = WPAD_Data( chan );
|
WPADData *data = WPAD_Data(chan);
|
||||||
|
|
||||||
switch ( data->exp.type )
|
switch (data->exp.type)
|
||||||
{
|
{
|
||||||
case WPAD_EXP_NUNCHUK:
|
case WPAD_EXP_NUNCHUK:
|
||||||
case WPAD_EXP_GUITARHERO3:
|
case WPAD_EXP_GUITARHERO3:
|
||||||
if ( right == 0 )
|
if (right == 0)
|
||||||
{
|
{
|
||||||
mag = data->exp.nunchuk.js.mag;
|
mag = data->exp.nunchuk.js.mag;
|
||||||
ang = data->exp.nunchuk.js.ang;
|
ang = data->exp.nunchuk.js.ang;
|
||||||
@ -134,7 +132,7 @@ s8 WPAD_Stick( u8 chan, u8 right, int axis )
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case WPAD_EXP_CLASSIC:
|
case WPAD_EXP_CLASSIC:
|
||||||
if ( right == 0 )
|
if (right == 0)
|
||||||
{
|
{
|
||||||
mag = data->exp.classic.ljs.mag;
|
mag = data->exp.classic.ljs.mag;
|
||||||
ang = data->exp.classic.ljs.ang;
|
ang = data->exp.classic.ljs.ang;
|
||||||
@ -151,14 +149,15 @@ s8 WPAD_Stick( u8 chan, u8 right, int axis )
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* calculate x/y value (angle need to be converted into radian) */
|
/* calculate x/y value (angle need to be converted into radian) */
|
||||||
if ( mag > 1.0 ) mag = 1.0;
|
if (mag > 1.0)
|
||||||
else if ( mag < -1.0 ) mag = -1.0;
|
mag = 1.0;
|
||||||
|
else if (mag < -1.0) mag = -1.0;
|
||||||
double val;
|
double val;
|
||||||
|
|
||||||
if ( axis == 0 ) // x-axis
|
if (axis == 0) // x-axis
|
||||||
val = mag * sin( ( PI * ang ) / 180.0f );
|
val = mag * sin((PI * ang) / 180.0f);
|
||||||
else // y-axis
|
else // y-axis
|
||||||
val = mag * cos( ( PI * ang ) / 180.0f );
|
val = mag * cos((PI * ang) / 180.0f);
|
||||||
|
|
||||||
return ( s8 )( val * 128.0f );
|
return (s8) (val * 128.0f);
|
||||||
}
|
}
|
||||||
|
@ -20,6 +20,6 @@ extern int rumbleRequest[4];
|
|||||||
void SetupPads();
|
void SetupPads();
|
||||||
void UpdatePads();
|
void UpdatePads();
|
||||||
void ShutoffRumble();
|
void ShutoffRumble();
|
||||||
void DoRumble( int i );
|
void DoRumble(int i);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -18,48 +18,49 @@ int updateLanguageFiles()
|
|||||||
char languageFiles[50][MAXLANGUAGEFILES];
|
char languageFiles[50][MAXLANGUAGEFILES];
|
||||||
|
|
||||||
//get all the files in the language path
|
//get all the files in the language path
|
||||||
int countfiles = GetAllDirFiles( Settings.languagefiles_path );
|
int countfiles = GetAllDirFiles(Settings.languagefiles_path);
|
||||||
|
|
||||||
//give up now if we didn't find any
|
//give up now if we didn't find any
|
||||||
if ( !countfiles ) return -2;
|
if (!countfiles) return -2;
|
||||||
|
|
||||||
//now from the files we got, get only the .lang files
|
//now from the files we got, get only the .lang files
|
||||||
for ( int cnt = 0; cnt < countfiles; cnt++ )
|
for (int cnt = 0; cnt < countfiles; cnt++)
|
||||||
{
|
{
|
||||||
char filename[64];
|
char filename[64];
|
||||||
strlcpy( filename, GetFileName( cnt ), sizeof( filename ) );
|
strlcpy(filename, GetFileName(cnt), sizeof(filename));
|
||||||
if ( strcasestr( filename, ".lang" ) )
|
if (strcasestr(filename, ".lang"))
|
||||||
{
|
{
|
||||||
strcpy( languageFiles[cnt], filename );
|
strcpy(languageFiles[cnt], filename);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
subfoldercreate( Settings.languagefiles_path );
|
subfoldercreate(Settings.languagefiles_path);
|
||||||
|
|
||||||
//we assume that the network will already be init by another function
|
//we assume that the network will already be init by another function
|
||||||
// ( that has gui eletents in it because this one doesn't)
|
// ( that has gui eletents in it because this one doesn't)
|
||||||
int done = 0, j = 0;
|
int done = 0, j = 0;
|
||||||
if ( IsNetworkInit() )
|
if (IsNetworkInit())
|
||||||
{
|
{
|
||||||
//build the URL, save path, and download each file and save it
|
//build the URL, save path, and download each file and save it
|
||||||
while ( j < countfiles )
|
while (j < countfiles)
|
||||||
{
|
{
|
||||||
char savepath[150];
|
char savepath[150];
|
||||||
char codeurl[200];
|
char codeurl[200];
|
||||||
snprintf( codeurl, sizeof( codeurl ), "http://usbloader-gui.googlecode.com/svn/trunk/Languages/%s", languageFiles[j] );
|
snprintf(codeurl, sizeof(codeurl), "http://usbloader-gui.googlecode.com/svn/trunk/Languages/%s",
|
||||||
snprintf( savepath, sizeof( savepath ), "%s%s", Settings.languagefiles_path, languageFiles[j] );
|
languageFiles[j]);
|
||||||
|
snprintf(savepath, sizeof(savepath), "%s%s", Settings.languagefiles_path, languageFiles[j]);
|
||||||
|
|
||||||
struct block file = downloadfile( codeurl );
|
struct block file = downloadfile(codeurl);
|
||||||
|
|
||||||
if ( file.data != NULL )
|
if (file.data != NULL)
|
||||||
{
|
{
|
||||||
FILE * pfile;
|
FILE * pfile;
|
||||||
pfile = fopen( savepath, "wb" );
|
pfile = fopen(savepath, "wb");
|
||||||
if ( pfile != NULL )
|
if (pfile != NULL)
|
||||||
{
|
{
|
||||||
fwrite( file.data, 1, file.size, pfile );
|
fwrite(file.data, 1, file.size, pfile);
|
||||||
fclose( pfile );
|
fclose(pfile);
|
||||||
free( file.data );
|
free(file.data);
|
||||||
done++;
|
done++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -75,5 +76,3 @@ int updateLanguageFiles()
|
|||||||
return done;
|
return done;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -12,28 +12,26 @@ typedef struct _MSG
|
|||||||
} MSG;
|
} MSG;
|
||||||
static MSG *baseMSG = 0;
|
static MSG *baseMSG = 0;
|
||||||
|
|
||||||
|
|
||||||
#define HASHWORDBITS 32
|
#define HASHWORDBITS 32
|
||||||
|
|
||||||
/* Defines the so called `hashpjw' function by P.J. Weinberger
|
/* Defines the so called `hashpjw' function by P.J. Weinberger
|
||||||
[see Aho/Sethi/Ullman, COMPILERS: Principles, Techniques and Tools,
|
[see Aho/Sethi/Ullman, COMPILERS: Principles, Techniques and Tools,
|
||||||
1986, 1987 Bell Telephone Laboratories, Inc.] */
|
1986, 1987 Bell Telephone Laboratories, Inc.] */
|
||||||
static inline u32
|
static inline u32 hash_string(const char *str_param)
|
||||||
hash_string ( const char *str_param )
|
|
||||||
{
|
{
|
||||||
u32 hval, g;
|
u32 hval, g;
|
||||||
const char *str = str_param;
|
const char *str = str_param;
|
||||||
|
|
||||||
/* Compute the hash value for the given string. */
|
/* Compute the hash value for the given string. */
|
||||||
hval = 0;
|
hval = 0;
|
||||||
while ( *str != '\0' )
|
while (*str != '\0')
|
||||||
{
|
{
|
||||||
hval <<= 4;
|
hval <<= 4;
|
||||||
hval += ( u8 ) * str++;
|
hval += (u8) *str++;
|
||||||
g = hval & ( ( u32 ) 0xf << ( HASHWORDBITS - 4 ) );
|
g = hval & ((u32) 0xf << (HASHWORDBITS - 4));
|
||||||
if ( g != 0 )
|
if (g != 0)
|
||||||
{
|
{
|
||||||
hval ^= g >> ( HASHWORDBITS - 8 );
|
hval ^= g >> (HASHWORDBITS - 8);
|
||||||
hval ^= g;
|
hval ^= g;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -42,54 +40,53 @@ hash_string ( const char *str_param )
|
|||||||
|
|
||||||
/* Expand some escape sequences found in the argument string. */
|
/* Expand some escape sequences found in the argument string. */
|
||||||
static char *
|
static char *
|
||||||
expand_escape ( const char *str )
|
expand_escape(const char *str)
|
||||||
{
|
{
|
||||||
char *retval, *rp;
|
char *retval, *rp;
|
||||||
const char *cp = str;
|
const char *cp = str;
|
||||||
|
|
||||||
retval = ( char * ) malloc ( strlen ( str ) + 1 );
|
retval = (char *) malloc(strlen(str) + 1);
|
||||||
if ( retval == NULL ) return NULL;
|
if (retval == NULL) return NULL;
|
||||||
rp = retval;
|
rp = retval;
|
||||||
|
|
||||||
while ( cp[0] != '\0' && cp[0] != '\\' )
|
while (cp[0] != '\0' && cp[0] != '\\')
|
||||||
*rp++ = *cp++;
|
*rp++ = *cp++;
|
||||||
if ( cp[0] == '\0' )
|
if (cp[0] == '\0') goto terminate;
|
||||||
goto terminate;
|
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
|
|
||||||
/* Here cp[0] == '\\'. */
|
/* Here cp[0] == '\\'. */
|
||||||
switch ( *++cp )
|
switch (*++cp)
|
||||||
{
|
{
|
||||||
case '\"': /* " */
|
case '\"': /* " */
|
||||||
*rp++ = '\"';
|
*rp++ = '\"';
|
||||||
++cp;
|
++cp;
|
||||||
break;
|
break;
|
||||||
case 'a': /* alert */
|
case 'a': /* alert */
|
||||||
*rp++ = '\a';
|
*rp++ = '\a';
|
||||||
++cp;
|
++cp;
|
||||||
break;
|
break;
|
||||||
case 'b': /* backspace */
|
case 'b': /* backspace */
|
||||||
*rp++ = '\b';
|
*rp++ = '\b';
|
||||||
++cp;
|
++cp;
|
||||||
break;
|
break;
|
||||||
case 'f': /* form feed */
|
case 'f': /* form feed */
|
||||||
*rp++ = '\f';
|
*rp++ = '\f';
|
||||||
++cp;
|
++cp;
|
||||||
break;
|
break;
|
||||||
case 'n': /* new line */
|
case 'n': /* new line */
|
||||||
*rp++ = '\n';
|
*rp++ = '\n';
|
||||||
++cp;
|
++cp;
|
||||||
break;
|
break;
|
||||||
case 'r': /* carriage return */
|
case 'r': /* carriage return */
|
||||||
*rp++ = '\r';
|
*rp++ = '\r';
|
||||||
++cp;
|
++cp;
|
||||||
break;
|
break;
|
||||||
case 't': /* horizontal tab */
|
case 't': /* horizontal tab */
|
||||||
*rp++ = '\t';
|
*rp++ = '\t';
|
||||||
++cp;
|
++cp;
|
||||||
break;
|
break;
|
||||||
case 'v': /* vertical tab */
|
case 'v': /* vertical tab */
|
||||||
*rp++ = '\v';
|
*rp++ = '\v';
|
||||||
++cp;
|
++cp;
|
||||||
break;
|
break;
|
||||||
@ -105,147 +102,139 @@ expand_escape ( const char *str )
|
|||||||
case '5':
|
case '5':
|
||||||
case '6':
|
case '6':
|
||||||
case '7':
|
case '7':
|
||||||
{
|
{
|
||||||
int ch = *cp++ - '0';
|
int ch = *cp++ - '0';
|
||||||
|
|
||||||
if ( *cp >= '0' && *cp <= '7' )
|
if (*cp >= '0' && *cp <= '7')
|
||||||
|
{
|
||||||
|
ch *= 8;
|
||||||
|
ch += *cp++ - '0';
|
||||||
|
|
||||||
|
if (*cp >= '0' && *cp <= '7')
|
||||||
{
|
{
|
||||||
ch *= 8;
|
ch *= 8;
|
||||||
ch += *cp++ - '0';
|
ch += *cp++ - '0';
|
||||||
|
|
||||||
if ( *cp >= '0' && *cp <= '7' )
|
|
||||||
{
|
|
||||||
ch *= 8;
|
|
||||||
ch += *cp++ - '0';
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
*rp = ch;
|
|
||||||
}
|
}
|
||||||
|
*rp = ch;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
*rp = '\\';
|
*rp = '\\';
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
while ( cp[0] != '\0' && cp[0] != '\\' )
|
while (cp[0] != '\0' && cp[0] != '\\')
|
||||||
*rp++ = *cp++;
|
*rp++ = *cp++;
|
||||||
}
|
} while (cp[0] != '\0');
|
||||||
while ( cp[0] != '\0' );
|
|
||||||
|
|
||||||
/* Terminate string. */
|
/* Terminate string. */
|
||||||
terminate:
|
terminate: *rp = '\0';
|
||||||
*rp = '\0';
|
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
static MSG *findMSG( u32 id )
|
static MSG *findMSG(u32 id)
|
||||||
{
|
{
|
||||||
MSG *msg;
|
MSG *msg;
|
||||||
for ( msg = baseMSG; msg; msg = msg->next )
|
for (msg = baseMSG; msg; msg = msg->next)
|
||||||
{
|
{
|
||||||
if ( msg->id == id )
|
if (msg->id == id) return msg;
|
||||||
return msg;
|
|
||||||
}
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static MSG *setMSG( const char *msgid, const char *msgstr )
|
static MSG *setMSG(const char *msgid, const char *msgstr)
|
||||||
{
|
{
|
||||||
u32 id = hash_string( msgid );
|
u32 id = hash_string(msgid);
|
||||||
MSG *msg = findMSG( id );
|
MSG *msg = findMSG(id);
|
||||||
if ( !msg )
|
if (!msg)
|
||||||
{
|
{
|
||||||
msg = ( MSG * )malloc( sizeof( MSG ) );
|
msg = (MSG *) malloc(sizeof(MSG));
|
||||||
msg->id = id;
|
msg->id = id;
|
||||||
msg->msgstr = NULL;
|
msg->msgstr = NULL;
|
||||||
msg->next = baseMSG;
|
msg->next = baseMSG;
|
||||||
baseMSG = msg;
|
baseMSG = msg;
|
||||||
}
|
}
|
||||||
if ( msg )
|
if (msg)
|
||||||
{
|
{
|
||||||
if ( msgstr )
|
if (msgstr)
|
||||||
{
|
{
|
||||||
if ( msg->msgstr ) free( msg->msgstr );
|
if (msg->msgstr) free(msg->msgstr);
|
||||||
//msg->msgstr = strdup(msgstr);
|
//msg->msgstr = strdup(msgstr);
|
||||||
msg->msgstr = expand_escape( msgstr );
|
msg->msgstr = expand_escape(msgstr);
|
||||||
}
|
}
|
||||||
return msg;
|
return msg;
|
||||||
}
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
void gettextCleanUp( void )
|
void gettextCleanUp(void)
|
||||||
{
|
{
|
||||||
while ( baseMSG )
|
while (baseMSG)
|
||||||
{
|
{
|
||||||
MSG *nextMsg = baseMSG->next;
|
MSG *nextMsg = baseMSG->next;
|
||||||
free( baseMSG->msgstr );
|
free(baseMSG->msgstr);
|
||||||
free( baseMSG );
|
free(baseMSG);
|
||||||
baseMSG = nextMsg;
|
baseMSG = nextMsg;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool gettextLoadLanguage(const char* langFile)
|
||||||
bool gettextLoadLanguage( const char* langFile )
|
|
||||||
{
|
{
|
||||||
FILE *f;
|
FILE *f;
|
||||||
char line[200];
|
char line[200];
|
||||||
char *lastID = NULL;
|
char *lastID = NULL;
|
||||||
|
|
||||||
gettextCleanUp();
|
gettextCleanUp();
|
||||||
f = fopen( langFile, "r" );
|
f = fopen(langFile, "r");
|
||||||
if ( !f )
|
if (!f) return false;
|
||||||
return false;
|
|
||||||
|
|
||||||
while ( fgets( line, sizeof( line ), f ) )
|
while (fgets(line, sizeof(line), f))
|
||||||
{
|
{
|
||||||
// lines starting with # are comments
|
// lines starting with # are comments
|
||||||
if ( line[0] == '#' )
|
if (line[0] == '#')
|
||||||
continue;
|
continue;
|
||||||
else if ( strncmp( line, "msgid \"", 7 ) == 0 )
|
else if (strncmp(line, "msgid \"", 7) == 0)
|
||||||
{
|
{
|
||||||
char *msgid, *end;
|
char *msgid, *end;
|
||||||
if ( lastID )
|
if (lastID)
|
||||||
{
|
{
|
||||||
free( lastID );
|
free(lastID);
|
||||||
lastID = NULL;
|
lastID = NULL;
|
||||||
}
|
}
|
||||||
msgid = &line[7];
|
msgid = &line[7];
|
||||||
end = strrchr( msgid, '"' );
|
end = strrchr(msgid, '"');
|
||||||
if ( end && end - msgid > 1 )
|
if (end && end - msgid > 1)
|
||||||
{
|
{
|
||||||
*end = 0;
|
*end = 0;
|
||||||
lastID = strdup( msgid );
|
lastID = strdup(msgid);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if ( strncmp( line, "msgstr \"", 8 ) == 0 )
|
else if (strncmp(line, "msgstr \"", 8) == 0)
|
||||||
{
|
{
|
||||||
char *msgstr, *end;
|
char *msgstr, *end;
|
||||||
|
|
||||||
if ( lastID == NULL )
|
if (lastID == NULL) continue;
|
||||||
continue;
|
|
||||||
|
|
||||||
msgstr = &line[8];
|
msgstr = &line[8];
|
||||||
end = strrchr( msgstr, '"' );
|
end = strrchr(msgstr, '"');
|
||||||
if ( end && end - msgstr > 1 )
|
if (end && end - msgstr > 1)
|
||||||
{
|
{
|
||||||
*end = 0;
|
*end = 0;
|
||||||
setMSG( lastID, msgstr );
|
setMSG(lastID, msgstr);
|
||||||
}
|
}
|
||||||
free( lastID );
|
free(lastID);
|
||||||
lastID = NULL;
|
lastID = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fclose( f );
|
fclose(f);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
const char *gettext( const char *msgid )
|
const char *gettext(const char *msgid)
|
||||||
{
|
{
|
||||||
MSG *msg = findMSG( hash_string( msgid ) );
|
MSG *msg = findMSG(hash_string(msgid));
|
||||||
if ( msg && msg->msgstr ) return msg->msgstr;
|
if (msg && msg->msgstr) return msg->msgstr;
|
||||||
return msgid;
|
return msgid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -6,18 +6,16 @@ extern "C"
|
|||||||
{
|
{
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
bool gettextLoadLanguage(const char* langFile);
|
||||||
bool gettextLoadLanguage( const char* langFile );
|
void gettextCleanUp(void);
|
||||||
void gettextCleanUp( void );
|
|
||||||
/*
|
/*
|
||||||
* input msg = a text in ASCII
|
* input msg = a text in ASCII
|
||||||
* output = the translated msg in utf-8
|
* output = the translated msg in utf-8
|
||||||
*/
|
*/
|
||||||
const char *gettext( const char *msg );
|
const char *gettext(const char *msg);
|
||||||
#define tr(s) gettext(s)
|
#define tr(s) gettext(s)
|
||||||
#define trNOOP(s) (s)
|
#define trNOOP(s) (s)
|
||||||
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -7,13 +7,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -24,7 +24,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __BIT_OPS_H
|
#ifndef __BIT_OPS_H
|
||||||
#define __BIT_OPS_H
|
#define __BIT_OPS_H
|
||||||
@ -32,30 +32,30 @@
|
|||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
/*-----------------------------------------------------------------
|
/*-----------------------------------------------------------------
|
||||||
Functions to deal with little endian values stored in uint8_t arrays
|
Functions to deal with little endian values stored in uint8_t arrays
|
||||||
-----------------------------------------------------------------*/
|
-----------------------------------------------------------------*/
|
||||||
static inline uint16_t u8array_to_u16 ( const uint8_t* item, int offset )
|
static inline uint16_t u8array_to_u16(const uint8_t* item, int offset)
|
||||||
{
|
{
|
||||||
return ( item[offset] | ( item[offset + 1] << 8 ) );
|
return (item[offset] | (item[offset + 1] << 8));
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline uint32_t u8array_to_u32 ( const uint8_t* item, int offset )
|
static inline uint32_t u8array_to_u32(const uint8_t* item, int offset)
|
||||||
{
|
{
|
||||||
return ( item[offset] | ( item[offset + 1] << 8 ) | ( item[offset + 2] << 16 ) | ( item[offset + 3] << 24 ) );
|
return (item[offset] | (item[offset + 1] << 8) | (item[offset + 2] << 16) | (item[offset + 3] << 24));
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void u16_to_u8array ( uint8_t* item, int offset, uint16_t value )
|
static inline void u16_to_u8array(uint8_t* item, int offset, uint16_t value)
|
||||||
{
|
{
|
||||||
item[offset] = ( uint8_t ) value;
|
item[offset] = (uint8_t) value;
|
||||||
item[offset + 1] = ( uint8_t )( value >> 8 );
|
item[offset + 1] = (uint8_t) (value >> 8);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void u32_to_u8array ( uint8_t* item, int offset, uint32_t value )
|
static inline void u32_to_u8array(uint8_t* item, int offset, uint32_t value)
|
||||||
{
|
{
|
||||||
item[offset] = ( uint8_t ) value;
|
item[offset] = (uint8_t) value;
|
||||||
item[offset + 1] = ( uint8_t )( value >> 8 );
|
item[offset + 1] = (uint8_t) (value >> 8);
|
||||||
item[offset + 2] = ( uint8_t )( value >> 16 );
|
item[offset + 2] = (uint8_t) (value >> 16);
|
||||||
item[offset + 3] = ( uint8_t )( value >> 24 );
|
item[offset + 3] = (uint8_t) (value >> 24);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif // _BIT_OPS_H
|
#endif // _BIT_OPS_H
|
||||||
|
@ -14,13 +14,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -31,7 +31,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef _CACHE_H
|
#ifndef _CACHE_H
|
||||||
#define _CACHE_H
|
#define _CACHE_H
|
||||||
@ -44,91 +44,93 @@
|
|||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
sec_t sector;
|
sec_t sector;
|
||||||
unsigned int count;
|
unsigned int count;
|
||||||
unsigned int last_access;
|
unsigned int last_access;
|
||||||
bool dirty;
|
bool dirty;
|
||||||
uint8_t* cache;
|
uint8_t* cache;
|
||||||
} CACHE_ENTRY;
|
} CACHE_ENTRY;
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
const DISC_INTERFACE* disc;
|
const DISC_INTERFACE* disc;
|
||||||
sec_t endOfPartition;
|
sec_t endOfPartition;
|
||||||
unsigned int numberOfPages;
|
unsigned int numberOfPages;
|
||||||
unsigned int sectorsPerPage;
|
unsigned int sectorsPerPage;
|
||||||
CACHE_ENTRY* cacheEntries;
|
CACHE_ENTRY* cacheEntries;
|
||||||
} CACHE;
|
} CACHE;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Read data from a sector in the cache
|
Read data from a sector in the cache
|
||||||
If the sector is not in the cache, it will be swapped in
|
If the sector is not in the cache, it will be swapped in
|
||||||
offset is the position to start reading from
|
offset is the position to start reading from
|
||||||
size is the amount of data to read
|
size is the amount of data to read
|
||||||
Precondition: offset + size <= BYTES_PER_READ
|
Precondition: offset + size <= BYTES_PER_READ
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_readPartialSector ( CACHE* cache, void* buffer, sec_t sector, unsigned int offset, size_t size );
|
bool _FAT_cache_readPartialSector(CACHE* cache, void* buffer, sec_t sector, unsigned int offset, size_t size);
|
||||||
|
|
||||||
bool _FAT_cache_readLittleEndianValue ( CACHE* cache, uint32_t *value, sec_t sector, unsigned int offset, int num_bytes );
|
bool _FAT_cache_readLittleEndianValue(CACHE* cache, uint32_t *value, sec_t sector, unsigned int offset, int num_bytes);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write data to a sector in the cache
|
Write data to a sector in the cache
|
||||||
If the sector is not in the cache, it will be swapped in.
|
If the sector is not in the cache, it will be swapped in.
|
||||||
When the sector is swapped out, the data will be written to the disc
|
When the sector is swapped out, the data will be written to the disc
|
||||||
offset is the position to start writing to
|
offset is the position to start writing to
|
||||||
size is the amount of data to write
|
size is the amount of data to write
|
||||||
Precondition: offset + size <= BYTES_PER_READ
|
Precondition: offset + size <= BYTES_PER_READ
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_writePartialSector ( CACHE* cache, const void* buffer, sec_t sector, unsigned int offset, size_t size );
|
bool _FAT_cache_writePartialSector(CACHE* cache, const void* buffer, sec_t sector, unsigned int offset, size_t size);
|
||||||
|
|
||||||
bool _FAT_cache_writeLittleEndianValue ( CACHE* cache, const uint32_t value, sec_t sector, unsigned int offset, int num_bytes );
|
bool _FAT_cache_writeLittleEndianValue(CACHE* cache, const uint32_t value, sec_t sector, unsigned int offset,
|
||||||
|
int num_bytes);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write data to a sector in the cache, zeroing the sector first
|
Write data to a sector in the cache, zeroing the sector first
|
||||||
If the sector is not in the cache, it will be swapped in.
|
If the sector is not in the cache, it will be swapped in.
|
||||||
When the sector is swapped out, the data will be written to the disc
|
When the sector is swapped out, the data will be written to the disc
|
||||||
offset is the position to start writing to
|
offset is the position to start writing to
|
||||||
size is the amount of data to write
|
size is the amount of data to write
|
||||||
Precondition: offset + size <= BYTES_PER_READ
|
Precondition: offset + size <= BYTES_PER_READ
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_eraseWritePartialSector ( CACHE* cache, const void* buffer, sec_t sector, unsigned int offset, size_t size );
|
bool _FAT_cache_eraseWritePartialSector(CACHE* cache, const void* buffer, sec_t sector, unsigned int offset,
|
||||||
|
size_t size);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Read several sectors from the cache
|
Read several sectors from the cache
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_readSectors ( CACHE* cache, sec_t sector, sec_t numSectors, void* buffer );
|
bool _FAT_cache_readSectors(CACHE* cache, sec_t sector, sec_t numSectors, void* buffer);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Read a full sector from the cache
|
Read a full sector from the cache
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_cache_readSector ( CACHE* cache, void* buffer, sec_t sector )
|
static inline bool _FAT_cache_readSector(CACHE* cache, void* buffer, sec_t sector)
|
||||||
{
|
{
|
||||||
return _FAT_cache_readPartialSector ( cache, buffer, sector, 0, BYTES_PER_READ );
|
return _FAT_cache_readPartialSector(cache, buffer, sector, 0, BYTES_PER_READ);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write a full sector to the cache
|
Write a full sector to the cache
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_cache_writeSector ( CACHE* cache, const void* buffer, sec_t sector )
|
static inline bool _FAT_cache_writeSector(CACHE* cache, const void* buffer, sec_t sector)
|
||||||
{
|
{
|
||||||
return _FAT_cache_writePartialSector ( cache, buffer, sector, 0, BYTES_PER_READ );
|
return _FAT_cache_writePartialSector(cache, buffer, sector, 0, BYTES_PER_READ);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool _FAT_cache_writeSectors ( CACHE* cache, sec_t sector, sec_t numSectors, const void* buffer );
|
bool _FAT_cache_writeSectors(CACHE* cache, sec_t sector, sec_t numSectors, const void* buffer);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write any dirty sectors back to disc and clear out the contents of the cache
|
Write any dirty sectors back to disc and clear out the contents of the cache
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_flush ( CACHE* cache );
|
bool _FAT_cache_flush(CACHE* cache);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Clear out the contents of the cache without writing any dirty sectors first
|
Clear out the contents of the cache without writing any dirty sectors first
|
||||||
*/
|
*/
|
||||||
void _FAT_cache_invalidate ( CACHE* cache );
|
void _FAT_cache_invalidate(CACHE* cache);
|
||||||
|
|
||||||
CACHE* _FAT_cache_constructor ( unsigned int numberOfPages, unsigned int sectorsPerPage, const DISC_INTERFACE* discInterface, sec_t endOfPartition );
|
CACHE* _FAT_cache_constructor(unsigned int numberOfPages, unsigned int sectorsPerPage,
|
||||||
|
const DISC_INTERFACE* discInterface, sec_t endOfPartition);
|
||||||
|
|
||||||
void _FAT_cache_destructor ( CACHE* cache );
|
void _FAT_cache_destructor(CACHE* cache);
|
||||||
|
|
||||||
#endif // _CACHE_H
|
#endif // _CACHE_H
|
||||||
|
|
||||||
|
@ -7,13 +7,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -24,7 +24,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __COMMON_H
|
#ifndef __COMMON_H
|
||||||
#define __COMMON_H
|
#define __COMMON_H
|
||||||
@ -34,7 +34,6 @@
|
|||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
|
|
||||||
// Platform specific includes
|
// Platform specific includes
|
||||||
#include <gctypes.h>
|
#include <gctypes.h>
|
||||||
#include <ogc/disc_io.h>
|
#include <ogc/disc_io.h>
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -8,13 +8,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -25,7 +25,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __DIRECTORY_H
|
#ifndef __DIRECTORY_H
|
||||||
#define __DIRECTORY_H
|
#define __DIRECTORY_H
|
||||||
@ -56,22 +56,24 @@
|
|||||||
#define ATTRIB_SYS 0x04 // System
|
#define ATTRIB_SYS 0x04 // System
|
||||||
#define ATTRIB_HID 0x02 // Hidden
|
#define ATTRIB_HID 0x02 // Hidden
|
||||||
#define ATTRIB_RO 0x01 // Read only
|
#define ATTRIB_RO 0x01 // Read only
|
||||||
|
typedef enum
|
||||||
typedef enum {FT_DIRECTORY, FT_FILE} FILE_TYPE;
|
{
|
||||||
|
FT_DIRECTORY, FT_FILE
|
||||||
|
} FILE_TYPE;
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
uint32_t cluster;
|
uint32_t cluster;
|
||||||
sec_t sector;
|
sec_t sector;
|
||||||
int32_t offset;
|
int32_t offset;
|
||||||
} DIR_ENTRY_POSITION;
|
} DIR_ENTRY_POSITION;
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
uint8_t entryData[DIR_ENTRY_DATA_SIZE];
|
uint8_t entryData[DIR_ENTRY_DATA_SIZE];
|
||||||
DIR_ENTRY_POSITION dataStart; // Points to the start of the LFN entries of a file, or the alias for no LFN
|
DIR_ENTRY_POSITION dataStart; // Points to the start of the LFN entries of a file, or the alias for no LFN
|
||||||
DIR_ENTRY_POSITION dataEnd; // Always points to the file/directory's alias entry
|
DIR_ENTRY_POSITION dataEnd; // Always points to the file/directory's alias entry
|
||||||
char filename[MAX_FILENAME_LENGTH];
|
char filename[MAX_FILENAME_LENGTH];
|
||||||
} DIR_ENTRY;
|
} DIR_ENTRY;
|
||||||
|
|
||||||
// Directory entry offsets
|
// Directory entry offsets
|
||||||
@ -93,87 +95,87 @@ enum DIR_ENTRY_offset
|
|||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Returns true if the file specified by entry is a directory
|
Returns true if the file specified by entry is a directory
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_directory_isDirectory ( DIR_ENTRY* entry )
|
static inline bool _FAT_directory_isDirectory(DIR_ENTRY* entry)
|
||||||
{
|
{
|
||||||
return ( ( entry->entryData[DIR_ENTRY_attributes] & ATTRIB_DIR ) != 0 );
|
return ((entry->entryData[DIR_ENTRY_attributes] & ATTRIB_DIR) != 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline bool _FAT_directory_isWritable ( DIR_ENTRY* entry )
|
static inline bool _FAT_directory_isWritable(DIR_ENTRY* entry)
|
||||||
{
|
{
|
||||||
return ( ( entry->entryData[DIR_ENTRY_attributes] & ATTRIB_RO ) == 0 );
|
return ((entry->entryData[DIR_ENTRY_attributes] & ATTRIB_RO) == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline bool _FAT_directory_isDot ( DIR_ENTRY* entry )
|
static inline bool _FAT_directory_isDot(DIR_ENTRY* entry)
|
||||||
{
|
{
|
||||||
return ( ( entry->filename[0] == '.' ) && ( ( entry->filename[1] == '\0' ) ||
|
return ((entry->filename[0] == '.') && ((entry->filename[1] == '\0') || ((entry->filename[1] == '.')
|
||||||
( ( entry->filename[1] == '.' ) && entry->filename[2] == '\0' ) ) );
|
&& entry->filename[2] == '\0')));
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Reads the first directory entry from the directory starting at dirCluster
|
Reads the first directory entry from the directory starting at dirCluster
|
||||||
Places result in entry
|
Places result in entry
|
||||||
entry will be destroyed even if no directory entry is found
|
entry will be destroyed even if no directory entry is found
|
||||||
Returns true on success, false on failure
|
Returns true on success, false on failure
|
||||||
*/
|
*/
|
||||||
bool _FAT_directory_getFirstEntry ( PARTITION* partition, DIR_ENTRY* entry, uint32_t dirCluster );
|
bool _FAT_directory_getFirstEntry(PARTITION* partition, DIR_ENTRY* entry, uint32_t dirCluster);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Reads the next directory entry after the one already pointed to by entry
|
Reads the next directory entry after the one already pointed to by entry
|
||||||
Places result in entry
|
Places result in entry
|
||||||
entry will be destroyed even if no directory entry is found
|
entry will be destroyed even if no directory entry is found
|
||||||
Returns true on success, false on failure
|
Returns true on success, false on failure
|
||||||
*/
|
*/
|
||||||
bool _FAT_directory_getNextEntry ( PARTITION* partition, DIR_ENTRY* entry );
|
bool _FAT_directory_getNextEntry(PARTITION* partition, DIR_ENTRY* entry);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Gets the directory entry corrsponding to the supplied path
|
Gets the directory entry corrsponding to the supplied path
|
||||||
entry will be destroyed even if no directory entry is found
|
entry will be destroyed even if no directory entry is found
|
||||||
pathEnd specifies the end of the path string, for cutting strings short if needed
|
pathEnd specifies the end of the path string, for cutting strings short if needed
|
||||||
specify NULL to use the full length of path
|
specify NULL to use the full length of path
|
||||||
pathEnd is only a suggestion, and the path string will be searched up until the next PATH_SEPARATOR
|
pathEnd is only a suggestion, and the path string will be searched up until the next PATH_SEPARATOR
|
||||||
after pathEND.
|
after pathEND.
|
||||||
Returns true on success, false on failure
|
Returns true on success, false on failure
|
||||||
*/
|
*/
|
||||||
bool _FAT_directory_entryFromPath ( PARTITION* partition, DIR_ENTRY* entry, const char* path, const char* pathEnd );
|
bool _FAT_directory_entryFromPath(PARTITION* partition, DIR_ENTRY* entry, const char* path, const char* pathEnd);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Changes the current directory to the one specified by path
|
Changes the current directory to the one specified by path
|
||||||
Returns true on success, false on failure
|
Returns true on success, false on failure
|
||||||
*/
|
*/
|
||||||
bool _FAT_directory_chdir ( PARTITION* partition, const char* path );
|
bool _FAT_directory_chdir(PARTITION* partition, const char* path);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Removes the directory entry specified by entry
|
Removes the directory entry specified by entry
|
||||||
Assumes that entry is valid
|
Assumes that entry is valid
|
||||||
Returns true on success, false on failure
|
Returns true on success, false on failure
|
||||||
*/
|
*/
|
||||||
bool _FAT_directory_removeEntry ( PARTITION* partition, DIR_ENTRY* entry );
|
bool _FAT_directory_removeEntry(PARTITION* partition, DIR_ENTRY* entry);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Add a directory entry to the directory specified by dirCluster
|
Add a directory entry to the directory specified by dirCluster
|
||||||
The fileData, dataStart and dataEnd elements of the DIR_ENTRY struct are
|
The fileData, dataStart and dataEnd elements of the DIR_ENTRY struct are
|
||||||
updated with the new directory entry position and alias.
|
updated with the new directory entry position and alias.
|
||||||
Returns true on success, false on failure
|
Returns true on success, false on failure
|
||||||
*/
|
*/
|
||||||
bool _FAT_directory_addEntry ( PARTITION* partition, DIR_ENTRY* entry, uint32_t dirCluster );
|
bool _FAT_directory_addEntry(PARTITION* partition, DIR_ENTRY* entry, uint32_t dirCluster);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Get the start cluster of a file from it's entry data
|
Get the start cluster of a file from it's entry data
|
||||||
*/
|
*/
|
||||||
uint32_t _FAT_directory_entryGetCluster ( PARTITION* partition, const uint8_t* entryData );
|
uint32_t _FAT_directory_entryGetCluster(PARTITION* partition, const uint8_t* entryData);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Fill in the file name and entry data of DIR_ENTRY* entry.
|
Fill in the file name and entry data of DIR_ENTRY* entry.
|
||||||
Assumes that the entry's dataStart and dataEnd are correct
|
Assumes that the entry's dataStart and dataEnd are correct
|
||||||
Returns true on success, false on failure
|
Returns true on success, false on failure
|
||||||
*/
|
*/
|
||||||
bool _FAT_directory_entryFromPosition ( PARTITION* partition, DIR_ENTRY* entry );
|
bool _FAT_directory_entryFromPosition(PARTITION* partition, DIR_ENTRY* entry);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Fill in a stat struct based on a file entry
|
Fill in a stat struct based on a file entry
|
||||||
*/
|
*/
|
||||||
void _FAT_directory_entryStat ( PARTITION* partition, DIR_ENTRY* entry, struct stat *st );
|
void _FAT_directory_entryStat(PARTITION* partition, DIR_ENTRY* entry, struct stat *st);
|
||||||
|
|
||||||
#endif // _DIRECTORY_H
|
#endif // _DIRECTORY_H
|
||||||
|
@ -8,13 +8,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -25,93 +25,94 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
#ifndef _DISC_H
|
#ifndef _DISC_H
|
||||||
#define _DISC_H
|
#define _DISC_H
|
||||||
|
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
|
||||||
/*
|
/*
|
||||||
A list of all default devices to try at startup,
|
A list of all default devices to try at startup,
|
||||||
terminated by a {NULL,NULL} entry.
|
terminated by a {NULL,NULL} entry.
|
||||||
*/
|
*/
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
const char* name;
|
const char* name;
|
||||||
const DISC_INTERFACE* ( *getInterface )( void );
|
const DISC_INTERFACE* (*getInterface)(void);
|
||||||
} INTERFACE_ID;
|
} INTERFACE_ID;
|
||||||
extern const INTERFACE_ID _FAT_disc_interfaces[];
|
extern const INTERFACE_ID _FAT_disc_interfaces[];
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Check if a disc is inserted
|
Check if a disc is inserted
|
||||||
Return true if a disc is inserted and ready, false otherwise
|
Return true if a disc is inserted and ready, false otherwise
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_disc_isInserted ( const DISC_INTERFACE* disc )
|
static inline bool _FAT_disc_isInserted(const DISC_INTERFACE* disc)
|
||||||
{
|
{
|
||||||
return disc->isInserted();
|
return disc->isInserted();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Read numSectors sectors from a disc, starting at sector.
|
Read numSectors sectors from a disc, starting at sector.
|
||||||
numSectors is between 1 and LIMIT_SECTORS if LIMIT_SECTORS is defined,
|
numSectors is between 1 and LIMIT_SECTORS if LIMIT_SECTORS is defined,
|
||||||
else it is at least 1
|
else it is at least 1
|
||||||
sector is 0 or greater
|
sector is 0 or greater
|
||||||
buffer is a pointer to the memory to fill
|
buffer is a pointer to the memory to fill
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_disc_readSectors ( const DISC_INTERFACE* disc, sec_t sector, sec_t numSectors, void* buffer )
|
static inline bool _FAT_disc_readSectors(const DISC_INTERFACE* disc, sec_t sector, sec_t numSectors, void* buffer)
|
||||||
{
|
{
|
||||||
return disc->readSectors ( sector, numSectors, buffer );
|
return disc->readSectors(sector, numSectors, buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write numSectors sectors to a disc, starting at sector.
|
Write numSectors sectors to a disc, starting at sector.
|
||||||
numSectors is between 1 and LIMIT_SECTORS if LIMIT_SECTORS is defined,
|
numSectors is between 1 and LIMIT_SECTORS if LIMIT_SECTORS is defined,
|
||||||
else it is at least 1
|
else it is at least 1
|
||||||
sector is 0 or greater
|
sector is 0 or greater
|
||||||
buffer is a pointer to the memory to read from
|
buffer is a pointer to the memory to read from
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_disc_writeSectors ( const DISC_INTERFACE* disc, sec_t sector, sec_t numSectors, const void* buffer )
|
static inline bool _FAT_disc_writeSectors(const DISC_INTERFACE* disc, sec_t sector, sec_t numSectors,
|
||||||
|
const void* buffer)
|
||||||
{
|
{
|
||||||
return disc->writeSectors ( sector, numSectors, buffer );
|
return disc->writeSectors(sector, numSectors, buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Reset the card back to a ready state
|
Reset the card back to a ready state
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_disc_clearStatus ( const DISC_INTERFACE* disc )
|
static inline bool _FAT_disc_clearStatus(const DISC_INTERFACE* disc)
|
||||||
{
|
{
|
||||||
return disc->clearStatus();
|
return disc->clearStatus();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Initialise the disc to a state ready for data reading or writing
|
Initialise the disc to a state ready for data reading or writing
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_disc_startup ( const DISC_INTERFACE* disc )
|
static inline bool _FAT_disc_startup(const DISC_INTERFACE* disc)
|
||||||
{
|
{
|
||||||
return disc->startup();
|
return disc->startup();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Put the disc in a state ready for power down.
|
Put the disc in a state ready for power down.
|
||||||
Complete any pending writes and disable the disc if necessary
|
Complete any pending writes and disable the disc if necessary
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_disc_shutdown ( const DISC_INTERFACE* disc )
|
static inline bool _FAT_disc_shutdown(const DISC_INTERFACE* disc)
|
||||||
{
|
{
|
||||||
return disc->shutdown();
|
return disc->shutdown();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Return a 32 bit value unique to each type of interface
|
Return a 32 bit value unique to each type of interface
|
||||||
*/
|
*/
|
||||||
static inline uint32_t _FAT_disc_hostType ( const DISC_INTERFACE* disc )
|
static inline uint32_t _FAT_disc_hostType(const DISC_INTERFACE* disc)
|
||||||
{
|
{
|
||||||
return disc->ioType;
|
return disc->ioType;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Return a 32 bit value that specifies the capabilities of the disc
|
Return a 32 bit value that specifies the capabilities of the disc
|
||||||
*/
|
*/
|
||||||
static inline uint32_t _FAT_disc_features ( const DISC_INTERFACE* disc )
|
static inline uint32_t _FAT_disc_features(const DISC_INTERFACE* disc)
|
||||||
{
|
{
|
||||||
return disc->features;
|
return disc->features;
|
||||||
}
|
}
|
||||||
|
@ -8,13 +8,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -25,48 +25,42 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "disc_fat.h"
|
#include "disc_fat.h"
|
||||||
|
|
||||||
/*
|
/*
|
||||||
The list of interfaces consists of a series of name/interface pairs.
|
The list of interfaces consists of a series of name/interface pairs.
|
||||||
The interface is returned via a simple function. This allows for
|
The interface is returned via a simple function. This allows for
|
||||||
platforms where the interface has to be "assembled" before it can
|
platforms where the interface has to be "assembled" before it can
|
||||||
be used, like DLDI on the NDS. For cases where a simple struct
|
be used, like DLDI on the NDS. For cases where a simple struct
|
||||||
is available, wrapper functions are used.
|
is available, wrapper functions are used.
|
||||||
The list is terminated by a NULL/NULL entry.
|
The list is terminated by a NULL/NULL entry.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* ====================== Wii ====================== */
|
/* ====================== Wii ====================== */
|
||||||
#include <sdcard/wiisd_io.h>
|
#include <sdcard/wiisd_io.h>
|
||||||
#include "usbloader/usbstorage2.h"
|
#include "usbloader/usbstorage2.h"
|
||||||
#include <sdcard/gcsd.h>
|
#include <sdcard/gcsd.h>
|
||||||
|
|
||||||
static const DISC_INTERFACE* get_io_wiisd ( void )
|
static const DISC_INTERFACE* get_io_wiisd(void)
|
||||||
{
|
{
|
||||||
return &__io_wiisd;
|
return &__io_wiisd;
|
||||||
}
|
}
|
||||||
static const DISC_INTERFACE* get_io_usbstorage ( void )
|
static const DISC_INTERFACE* get_io_usbstorage(void)
|
||||||
{
|
{
|
||||||
return &__io_usbstorage2;
|
return &__io_usbstorage2;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const DISC_INTERFACE* get_io_gcsda ( void )
|
static const DISC_INTERFACE* get_io_gcsda(void)
|
||||||
{
|
{
|
||||||
return &__io_gcsda;
|
return &__io_gcsda;
|
||||||
}
|
}
|
||||||
static const DISC_INTERFACE* get_io_gcsdb ( void )
|
static const DISC_INTERFACE* get_io_gcsdb(void)
|
||||||
{
|
{
|
||||||
return &__io_gcsdb;
|
return &__io_gcsdb;
|
||||||
}
|
}
|
||||||
|
|
||||||
const INTERFACE_ID _FAT_disc_interfaces[] =
|
const INTERFACE_ID _FAT_disc_interfaces[] = { { "sd", get_io_wiisd }, { "usb", get_io_usbstorage }, { "carda",
|
||||||
{
|
get_io_gcsda }, { "cardb", get_io_gcsdb }, { NULL, NULL } };
|
||||||
{"sd", get_io_wiisd},
|
|
||||||
{"usb", get_io_usbstorage},
|
|
||||||
{"carda", get_io_gcsda},
|
|
||||||
{"cardb", get_io_gcsdb},
|
|
||||||
{NULL, NULL}
|
|
||||||
};
|
|
||||||
|
|
||||||
|
@ -8,13 +8,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -25,93 +25,94 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
#ifndef __DISC_H
|
#ifndef __DISC_H
|
||||||
#define __DISC_H
|
#define __DISC_H
|
||||||
|
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
|
||||||
/*
|
/*
|
||||||
A list of all default devices to try at startup,
|
A list of all default devices to try at startup,
|
||||||
terminated by a {NULL,NULL} entry.
|
terminated by a {NULL,NULL} entry.
|
||||||
*/
|
*/
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
const char* name;
|
const char* name;
|
||||||
const DISC_INTERFACE* ( *getInterface )( void );
|
const DISC_INTERFACE* (*getInterface)(void);
|
||||||
} INTERFACE_ID;
|
} INTERFACE_ID;
|
||||||
extern const INTERFACE_ID _FAT_disc_interfaces[];
|
extern const INTERFACE_ID _FAT_disc_interfaces[];
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Check if a disc is inserted
|
Check if a disc is inserted
|
||||||
Return true if a disc is inserted and ready, false otherwise
|
Return true if a disc is inserted and ready, false otherwise
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_disc_isInserted ( const DISC_INTERFACE* disc )
|
static inline bool _FAT_disc_isInserted(const DISC_INTERFACE* disc)
|
||||||
{
|
{
|
||||||
return disc->isInserted();
|
return disc->isInserted();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Read numSectors sectors from a disc, starting at sector.
|
Read numSectors sectors from a disc, starting at sector.
|
||||||
numSectors is between 1 and LIMIT_SECTORS if LIMIT_SECTORS is defined,
|
numSectors is between 1 and LIMIT_SECTORS if LIMIT_SECTORS is defined,
|
||||||
else it is at least 1
|
else it is at least 1
|
||||||
sector is 0 or greater
|
sector is 0 or greater
|
||||||
buffer is a pointer to the memory to fill
|
buffer is a pointer to the memory to fill
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_disc_readSectors ( const DISC_INTERFACE* disc, sec_t sector, sec_t numSectors, void* buffer )
|
static inline bool _FAT_disc_readSectors(const DISC_INTERFACE* disc, sec_t sector, sec_t numSectors, void* buffer)
|
||||||
{
|
{
|
||||||
return disc->readSectors ( sector, numSectors, buffer );
|
return disc->readSectors(sector, numSectors, buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write numSectors sectors to a disc, starting at sector.
|
Write numSectors sectors to a disc, starting at sector.
|
||||||
numSectors is between 1 and LIMIT_SECTORS if LIMIT_SECTORS is defined,
|
numSectors is between 1 and LIMIT_SECTORS if LIMIT_SECTORS is defined,
|
||||||
else it is at least 1
|
else it is at least 1
|
||||||
sector is 0 or greater
|
sector is 0 or greater
|
||||||
buffer is a pointer to the memory to read from
|
buffer is a pointer to the memory to read from
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_disc_writeSectors ( const DISC_INTERFACE* disc, sec_t sector, sec_t numSectors, const void* buffer )
|
static inline bool _FAT_disc_writeSectors(const DISC_INTERFACE* disc, sec_t sector, sec_t numSectors,
|
||||||
|
const void* buffer)
|
||||||
{
|
{
|
||||||
return disc->writeSectors ( sector, numSectors, buffer );
|
return disc->writeSectors(sector, numSectors, buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Reset the card back to a ready state
|
Reset the card back to a ready state
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_disc_clearStatus ( const DISC_INTERFACE* disc )
|
static inline bool _FAT_disc_clearStatus(const DISC_INTERFACE* disc)
|
||||||
{
|
{
|
||||||
return disc->clearStatus();
|
return disc->clearStatus();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Initialise the disc to a state ready for data reading or writing
|
Initialise the disc to a state ready for data reading or writing
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_disc_startup ( const DISC_INTERFACE* disc )
|
static inline bool _FAT_disc_startup(const DISC_INTERFACE* disc)
|
||||||
{
|
{
|
||||||
return disc->startup();
|
return disc->startup();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Put the disc in a state ready for power down.
|
Put the disc in a state ready for power down.
|
||||||
Complete any pending writes and disable the disc if necessary
|
Complete any pending writes and disable the disc if necessary
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_disc_shutdown ( const DISC_INTERFACE* disc )
|
static inline bool _FAT_disc_shutdown(const DISC_INTERFACE* disc)
|
||||||
{
|
{
|
||||||
return disc->shutdown();
|
return disc->shutdown();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Return a 32 bit value unique to each type of interface
|
Return a 32 bit value unique to each type of interface
|
||||||
*/
|
*/
|
||||||
static inline uint32_t _FAT_disc_hostType ( const DISC_INTERFACE* disc )
|
static inline uint32_t _FAT_disc_hostType(const DISC_INTERFACE* disc)
|
||||||
{
|
{
|
||||||
return disc->ioType;
|
return disc->ioType;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Return a 32 bit value that specifies the capabilities of the disc
|
Return a 32 bit value that specifies the capabilities of the disc
|
||||||
*/
|
*/
|
||||||
static inline uint32_t _FAT_disc_features ( const DISC_INTERFACE* disc )
|
static inline uint32_t _FAT_disc_features(const DISC_INTERFACE* disc)
|
||||||
{
|
{
|
||||||
return disc->features;
|
return disc->features;
|
||||||
}
|
}
|
||||||
|
@ -1,19 +1,19 @@
|
|||||||
/*
|
/*
|
||||||
fat.h
|
fat.h
|
||||||
Simple functionality for startup, mounting and unmounting of FAT-based devices.
|
Simple functionality for startup, mounting and unmounting of FAT-based devices.
|
||||||
|
|
||||||
Copyright (c) 2006 Michael "Chishm" Chisholm
|
Copyright (c) 2006 Michael "Chishm" Chisholm
|
||||||
|
|
||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -24,8 +24,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
#ifndef _LIBFAT_H
|
#ifndef _LIBFAT_H
|
||||||
#define _LIBFAT_H
|
#define _LIBFAT_H
|
||||||
@ -39,40 +38,41 @@ extern "C"
|
|||||||
#include <ogc/disc_io.h>
|
#include <ogc/disc_io.h>
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Initialise any inserted block-devices.
|
Initialise any inserted block-devices.
|
||||||
Add the fat device driver to the devoptab, making it available for standard file functions.
|
Add the fat device driver to the devoptab, making it available for standard file functions.
|
||||||
cacheSize: The number of pages to allocate for each inserted block-device
|
cacheSize: The number of pages to allocate for each inserted block-device
|
||||||
setAsDefaultDevice: if true, make this the default device driver for file operations
|
setAsDefaultDevice: if true, make this the default device driver for file operations
|
||||||
*/
|
*/
|
||||||
extern bool fatInit ( uint32_t cacheSize, bool setAsDefaultDevice );
|
extern bool fatInit(uint32_t cacheSize, bool setAsDefaultDevice);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Calls fatInit with setAsDefaultDevice = true and cacheSize optimised for the host system.
|
Calls fatInit with setAsDefaultDevice = true and cacheSize optimised for the host system.
|
||||||
*/
|
*/
|
||||||
extern bool fatInitDefault ( void );
|
extern bool fatInitDefault(void);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Mount the device pointed to by interface, and set up a devoptab entry for it as "name:".
|
Mount the device pointed to by interface, and set up a devoptab entry for it as "name:".
|
||||||
You can then access the filesystem using "name:/".
|
You can then access the filesystem using "name:/".
|
||||||
This will mount the active partition or the first valid partition on the disc,
|
This will mount the active partition or the first valid partition on the disc,
|
||||||
and will use a cache size optimized for the host system.
|
and will use a cache size optimized for the host system.
|
||||||
*/
|
*/
|
||||||
extern bool fatMountSimple ( const char* name, const DISC_INTERFACE* interface );
|
extern bool fatMountSimple(const char* name, const DISC_INTERFACE* interface);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Mount the device pointed to by interface, and set up a devoptab entry for it as "name:".
|
Mount the device pointed to by interface, and set up a devoptab entry for it as "name:".
|
||||||
You can then access the filesystem using "name:/".
|
You can then access the filesystem using "name:/".
|
||||||
If startSector = 0, it will mount the active partition of the first valid partition on
|
If startSector = 0, it will mount the active partition of the first valid partition on
|
||||||
the disc. Otherwise it will try to mount the partition starting at startSector.
|
the disc. Otherwise it will try to mount the partition starting at startSector.
|
||||||
cacheSize specifies the number of pages to allocate for the cache.
|
cacheSize specifies the number of pages to allocate for the cache.
|
||||||
This will not startup the disc, so you need to call interface->startup(); first.
|
This will not startup the disc, so you need to call interface->startup(); first.
|
||||||
*/
|
*/
|
||||||
extern bool fatMount ( const char* name, const DISC_INTERFACE* interface, sec_t startSector, uint32_t cacheSize, uint32_t SectorsPerPage );
|
extern bool fatMount(const char* name, const DISC_INTERFACE* interface, sec_t startSector, uint32_t cacheSize,
|
||||||
|
uint32_t SectorsPerPage);
|
||||||
/*
|
/*
|
||||||
Unmount the partition specified by name.
|
Unmount the partition specified by name.
|
||||||
If there are open files, it will attempt to synchronise them to disc.
|
If there are open files, it will attempt to synchronise them to disc.
|
||||||
*/
|
*/
|
||||||
extern void fatUnmount ( const char* name );
|
extern void fatUnmount(const char* name);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
|
@ -14,13 +14,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -31,7 +31,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
@ -46,24 +46,25 @@
|
|||||||
|
|
||||||
#define CACHE_FREE UINT_MAX
|
#define CACHE_FREE UINT_MAX
|
||||||
|
|
||||||
CACHE* _FAT_cache_constructor ( unsigned int numberOfPages, unsigned int sectorsPerPage, const DISC_INTERFACE* discInterface, sec_t endOfPartition )
|
CACHE* _FAT_cache_constructor(unsigned int numberOfPages, unsigned int sectorsPerPage,
|
||||||
|
const DISC_INTERFACE* discInterface, sec_t endOfPartition)
|
||||||
{
|
{
|
||||||
CACHE* cache;
|
CACHE* cache;
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
CACHE_ENTRY* cacheEntries;
|
CACHE_ENTRY* cacheEntries;
|
||||||
|
|
||||||
if ( numberOfPages < 2 )
|
if (numberOfPages < 2)
|
||||||
{
|
{
|
||||||
numberOfPages = 2;
|
numberOfPages = 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( sectorsPerPage < 8 )
|
if (sectorsPerPage < 8)
|
||||||
{
|
{
|
||||||
sectorsPerPage = 8;
|
sectorsPerPage = 8;
|
||||||
}
|
}
|
||||||
|
|
||||||
cache = ( CACHE* ) _FAT_mem_allocate ( sizeof( CACHE ) );
|
cache = (CACHE*) _FAT_mem_allocate(sizeof(CACHE));
|
||||||
if ( cache == NULL )
|
if (cache == NULL)
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
@ -73,21 +74,20 @@ CACHE* _FAT_cache_constructor ( unsigned int numberOfPages, unsigned int sectors
|
|||||||
cache->numberOfPages = numberOfPages;
|
cache->numberOfPages = numberOfPages;
|
||||||
cache->sectorsPerPage = sectorsPerPage;
|
cache->sectorsPerPage = sectorsPerPage;
|
||||||
|
|
||||||
|
cacheEntries = (CACHE_ENTRY*) _FAT_mem_allocate(sizeof(CACHE_ENTRY) * numberOfPages);
|
||||||
cacheEntries = ( CACHE_ENTRY* ) _FAT_mem_allocate ( sizeof( CACHE_ENTRY ) * numberOfPages );
|
if (cacheEntries == NULL)
|
||||||
if ( cacheEntries == NULL )
|
|
||||||
{
|
{
|
||||||
_FAT_mem_free ( cache );
|
_FAT_mem_free(cache);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
for ( i = 0; i < numberOfPages; i++ )
|
for (i = 0; i < numberOfPages; i++)
|
||||||
{
|
{
|
||||||
cacheEntries[i].sector = CACHE_FREE;
|
cacheEntries[i].sector = CACHE_FREE;
|
||||||
cacheEntries[i].count = 0;
|
cacheEntries[i].count = 0;
|
||||||
cacheEntries[i].last_access = 0;
|
cacheEntries[i].last_access = 0;
|
||||||
cacheEntries[i].dirty = false;
|
cacheEntries[i].dirty = false;
|
||||||
cacheEntries[i].cache = ( uint8_t* ) _FAT_mem_align ( sectorsPerPage * BYTES_PER_READ );
|
cacheEntries[i].cache = (uint8_t*) _FAT_mem_align(sectorsPerPage * BYTES_PER_READ);
|
||||||
}
|
}
|
||||||
|
|
||||||
cache->cacheEntries = cacheEntries;
|
cache->cacheEntries = cacheEntries;
|
||||||
@ -95,22 +95,21 @@ CACHE* _FAT_cache_constructor ( unsigned int numberOfPages, unsigned int sectors
|
|||||||
return cache;
|
return cache;
|
||||||
}
|
}
|
||||||
|
|
||||||
void _FAT_cache_destructor ( CACHE* cache )
|
void _FAT_cache_destructor(CACHE* cache)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
// Clear out cache before destroying it
|
// Clear out cache before destroying it
|
||||||
_FAT_cache_flush( cache );
|
_FAT_cache_flush(cache);
|
||||||
|
|
||||||
// Free memory in reverse allocation order
|
// Free memory in reverse allocation order
|
||||||
for ( i = 0; i < cache->numberOfPages; i++ )
|
for (i = 0; i < cache->numberOfPages; i++)
|
||||||
{
|
{
|
||||||
_FAT_mem_free ( cache->cacheEntries[i].cache );
|
_FAT_mem_free(cache->cacheEntries[i].cache);
|
||||||
}
|
}
|
||||||
_FAT_mem_free ( cache->cacheEntries );
|
_FAT_mem_free(cache->cacheEntries);
|
||||||
_FAT_mem_free ( cache );
|
_FAT_mem_free(cache);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static u32 accessCounter = 0;
|
static u32 accessCounter = 0;
|
||||||
|
|
||||||
static u32 accessTime()
|
static u32 accessTime()
|
||||||
@ -119,8 +118,7 @@ static u32 accessTime()
|
|||||||
return accessCounter;
|
return accessCounter;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static CACHE_ENTRY* _FAT_cache_getPage(CACHE *cache, sec_t sector)
|
||||||
static CACHE_ENTRY* _FAT_cache_getPage( CACHE *cache, sec_t sector )
|
|
||||||
{
|
{
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
CACHE_ENTRY* cacheEntries = cache->cacheEntries;
|
CACHE_ENTRY* cacheEntries = cache->cacheEntries;
|
||||||
@ -131,60 +129,61 @@ static CACHE_ENTRY* _FAT_cache_getPage( CACHE *cache, sec_t sector )
|
|||||||
unsigned int oldUsed = 0;
|
unsigned int oldUsed = 0;
|
||||||
unsigned int oldAccess = UINT_MAX;
|
unsigned int oldAccess = UINT_MAX;
|
||||||
|
|
||||||
for ( i = 0; i < numberOfPages; i++ )
|
for (i = 0; i < numberOfPages; i++)
|
||||||
{
|
{
|
||||||
if ( sector >= cacheEntries[i].sector && sector < ( cacheEntries[i].sector + cacheEntries[i].count ) )
|
if (sector >= cacheEntries[i].sector && sector < (cacheEntries[i].sector + cacheEntries[i].count))
|
||||||
{
|
{
|
||||||
cacheEntries[i].last_access = accessTime();
|
cacheEntries[i].last_access = accessTime();
|
||||||
return &( cacheEntries[i] );
|
return &(cacheEntries[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( foundFree == false && ( cacheEntries[i].sector == CACHE_FREE || cacheEntries[i].last_access < oldAccess ) )
|
if (foundFree == false && (cacheEntries[i].sector == CACHE_FREE || cacheEntries[i].last_access < oldAccess))
|
||||||
{
|
{
|
||||||
if ( cacheEntries[i].sector == CACHE_FREE ) foundFree = true;
|
if (cacheEntries[i].sector == CACHE_FREE) foundFree = true;
|
||||||
oldUsed = i;
|
oldUsed = i;
|
||||||
oldAccess = cacheEntries[i].last_access;
|
oldAccess = cacheEntries[i].last_access;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( foundFree == false && cacheEntries[oldUsed].dirty == true )
|
if (foundFree == false && cacheEntries[oldUsed].dirty == true)
|
||||||
{
|
{
|
||||||
if ( !_FAT_disc_writeSectors( cache->disc, cacheEntries[oldUsed].sector, cacheEntries[oldUsed].count, cacheEntries[oldUsed].cache ) ) return NULL;
|
if (!_FAT_disc_writeSectors(cache->disc, cacheEntries[oldUsed].sector, cacheEntries[oldUsed].count,
|
||||||
|
cacheEntries[oldUsed].cache)) return NULL;
|
||||||
cacheEntries[oldUsed].dirty = false;
|
cacheEntries[oldUsed].dirty = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
sector = ( sector / sectorsPerPage ) * sectorsPerPage; // align base sector to page size
|
sector = (sector / sectorsPerPage) * sectorsPerPage; // align base sector to page size
|
||||||
sec_t next_page = sector + sectorsPerPage;
|
sec_t next_page = sector + sectorsPerPage;
|
||||||
if ( next_page > cache->endOfPartition ) next_page = cache->endOfPartition;
|
if (next_page > cache->endOfPartition) next_page = cache->endOfPartition;
|
||||||
|
|
||||||
if ( !_FAT_disc_readSectors( cache->disc, sector, next_page - sector, cacheEntries[oldUsed].cache ) ) return NULL;
|
if (!_FAT_disc_readSectors(cache->disc, sector, next_page - sector, cacheEntries[oldUsed].cache)) return NULL;
|
||||||
|
|
||||||
cacheEntries[oldUsed].sector = sector;
|
cacheEntries[oldUsed].sector = sector;
|
||||||
cacheEntries[oldUsed].count = next_page - sector;
|
cacheEntries[oldUsed].count = next_page - sector;
|
||||||
cacheEntries[oldUsed].last_access = accessTime();
|
cacheEntries[oldUsed].last_access = accessTime();
|
||||||
|
|
||||||
return &( cacheEntries[oldUsed] );
|
return &(cacheEntries[oldUsed]);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool _FAT_cache_readSectors( CACHE *cache, sec_t sector, sec_t numSectors, void *buffer )
|
bool _FAT_cache_readSectors(CACHE *cache, sec_t sector, sec_t numSectors, void *buffer)
|
||||||
{
|
{
|
||||||
sec_t sec;
|
sec_t sec;
|
||||||
sec_t secs_to_read;
|
sec_t secs_to_read;
|
||||||
CACHE_ENTRY *entry;
|
CACHE_ENTRY *entry;
|
||||||
uint8_t *dest = buffer;
|
uint8_t *dest = buffer;
|
||||||
|
|
||||||
while ( numSectors > 0 )
|
while (numSectors > 0)
|
||||||
{
|
{
|
||||||
entry = _FAT_cache_getPage( cache, sector );
|
entry = _FAT_cache_getPage(cache, sector);
|
||||||
if ( entry == NULL ) return false;
|
if (entry == NULL) return false;
|
||||||
|
|
||||||
sec = sector - entry->sector;
|
sec = sector - entry->sector;
|
||||||
secs_to_read = entry->count - sec;
|
secs_to_read = entry->count - sec;
|
||||||
if ( secs_to_read > numSectors ) secs_to_read = numSectors;
|
if (secs_to_read > numSectors) secs_to_read = numSectors;
|
||||||
|
|
||||||
memcpy( dest, entry->cache + ( sec*BYTES_PER_READ ), ( secs_to_read*BYTES_PER_READ ) );
|
memcpy(dest, entry->cache + (sec * BYTES_PER_READ), (secs_to_read * BYTES_PER_READ));
|
||||||
|
|
||||||
dest += ( secs_to_read * BYTES_PER_READ );
|
dest += (secs_to_read * BYTES_PER_READ);
|
||||||
sector += secs_to_read;
|
sector += secs_to_read;
|
||||||
numSectors -= secs_to_read;
|
numSectors -= secs_to_read;
|
||||||
}
|
}
|
||||||
@ -193,111 +192,125 @@ bool _FAT_cache_readSectors( CACHE *cache, sec_t sector, sec_t numSectors, void
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Reads some data from a cache page, determined by the sector number
|
Reads some data from a cache page, determined by the sector number
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_readPartialSector ( CACHE* cache, void* buffer, sec_t sector, unsigned int offset, size_t size )
|
bool _FAT_cache_readPartialSector(CACHE* cache, void* buffer, sec_t sector, unsigned int offset, size_t size)
|
||||||
{
|
{
|
||||||
sec_t sec;
|
sec_t sec;
|
||||||
CACHE_ENTRY *entry;
|
CACHE_ENTRY *entry;
|
||||||
|
|
||||||
if ( offset + size > BYTES_PER_READ ) return false;
|
if (offset + size > BYTES_PER_READ) return false;
|
||||||
|
|
||||||
entry = _FAT_cache_getPage( cache, sector );
|
entry = _FAT_cache_getPage(cache, sector);
|
||||||
if ( entry == NULL ) return false;
|
if (entry == NULL) return false;
|
||||||
|
|
||||||
sec = sector - entry->sector;
|
sec = sector - entry->sector;
|
||||||
memcpy( buffer, entry->cache + ( ( sec*BYTES_PER_READ ) + offset ), size );
|
memcpy(buffer, entry->cache + ((sec * BYTES_PER_READ) + offset), size);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool _FAT_cache_readLittleEndianValue ( CACHE* cache, uint32_t *value, sec_t sector, unsigned int offset, int num_bytes )
|
bool _FAT_cache_readLittleEndianValue(CACHE* cache, uint32_t *value, sec_t sector, unsigned int offset, int num_bytes)
|
||||||
{
|
{
|
||||||
uint8_t buf[4];
|
uint8_t buf[4];
|
||||||
if ( !_FAT_cache_readPartialSector( cache, buf, sector, offset, num_bytes ) ) return false;
|
if (!_FAT_cache_readPartialSector(cache, buf, sector, offset, num_bytes)) return false;
|
||||||
|
|
||||||
switch ( num_bytes )
|
switch (num_bytes)
|
||||||
{
|
{
|
||||||
case 1: *value = buf[0]; break;
|
case 1:
|
||||||
case 2: *value = u8array_to_u16( buf, 0 ); break;
|
*value = buf[0];
|
||||||
case 4: *value = u8array_to_u32( buf, 0 ); break;
|
break;
|
||||||
default: return false;
|
case 2:
|
||||||
|
*value = u8array_to_u16(buf, 0);
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
*value = u8array_to_u32(buf, 0);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Writes some data to a cache page, making sure it is loaded into memory first.
|
Writes some data to a cache page, making sure it is loaded into memory first.
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_writePartialSector ( CACHE* cache, const void* buffer, sec_t sector, unsigned int offset, size_t size )
|
bool _FAT_cache_writePartialSector(CACHE* cache, const void* buffer, sec_t sector, unsigned int offset, size_t size)
|
||||||
{
|
{
|
||||||
sec_t sec;
|
sec_t sec;
|
||||||
CACHE_ENTRY *entry;
|
CACHE_ENTRY *entry;
|
||||||
|
|
||||||
if ( offset + size > BYTES_PER_READ ) return false;
|
if (offset + size > BYTES_PER_READ) return false;
|
||||||
|
|
||||||
entry = _FAT_cache_getPage( cache, sector );
|
entry = _FAT_cache_getPage(cache, sector);
|
||||||
if ( entry == NULL ) return false;
|
if (entry == NULL) return false;
|
||||||
|
|
||||||
sec = sector - entry->sector;
|
sec = sector - entry->sector;
|
||||||
memcpy( entry->cache + ( ( sec*BYTES_PER_READ ) + offset ), buffer, size );
|
memcpy(entry->cache + ((sec * BYTES_PER_READ) + offset), buffer, size);
|
||||||
|
|
||||||
entry->dirty = true;
|
entry->dirty = true;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool _FAT_cache_writeLittleEndianValue ( CACHE* cache, const uint32_t value, sec_t sector, unsigned int offset, int size )
|
bool _FAT_cache_writeLittleEndianValue(CACHE* cache, const uint32_t value, sec_t sector, unsigned int offset, int size)
|
||||||
{
|
{
|
||||||
uint8_t buf[4] = {0, 0, 0, 0};
|
uint8_t buf[4] = { 0, 0, 0, 0 };
|
||||||
|
|
||||||
switch ( size )
|
switch (size)
|
||||||
{
|
{
|
||||||
case 1: buf[0] = value; break;
|
case 1:
|
||||||
case 2: u16_to_u8array( buf, 0, value ); break;
|
buf[0] = value;
|
||||||
case 4: u32_to_u8array( buf, 0, value ); break;
|
break;
|
||||||
default: return false;
|
case 2:
|
||||||
|
u16_to_u8array(buf, 0, value);
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
u32_to_u8array(buf, 0, value);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return _FAT_cache_writePartialSector( cache, buf, sector, offset, size );
|
return _FAT_cache_writePartialSector(cache, buf, sector, offset, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Writes some data to a cache page, zeroing out the page first
|
Writes some data to a cache page, zeroing out the page first
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_eraseWritePartialSector ( CACHE* cache, const void* buffer, sec_t sector, unsigned int offset, size_t size )
|
bool _FAT_cache_eraseWritePartialSector(CACHE* cache, const void* buffer, sec_t sector, unsigned int offset,
|
||||||
|
size_t size)
|
||||||
{
|
{
|
||||||
sec_t sec;
|
sec_t sec;
|
||||||
CACHE_ENTRY *entry;
|
CACHE_ENTRY *entry;
|
||||||
|
|
||||||
if ( offset + size > BYTES_PER_READ ) return false;
|
if (offset + size > BYTES_PER_READ) return false;
|
||||||
|
|
||||||
entry = _FAT_cache_getPage( cache, sector );
|
entry = _FAT_cache_getPage(cache, sector);
|
||||||
if ( entry == NULL ) return false;
|
if (entry == NULL) return false;
|
||||||
|
|
||||||
sec = sector - entry->sector;
|
sec = sector - entry->sector;
|
||||||
memset( entry->cache + ( sec*BYTES_PER_READ ), 0, BYTES_PER_READ );
|
memset(entry->cache + (sec * BYTES_PER_READ), 0, BYTES_PER_READ);
|
||||||
memcpy( entry->cache + ( ( sec*BYTES_PER_READ ) + offset ), buffer, size );
|
memcpy(entry->cache + ((sec * BYTES_PER_READ) + offset), buffer, size);
|
||||||
|
|
||||||
entry->dirty = true;
|
entry->dirty = true;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static CACHE_ENTRY* _FAT_cache_findPage(CACHE *cache, sec_t sector, sec_t count)
|
||||||
static CACHE_ENTRY* _FAT_cache_findPage( CACHE *cache, sec_t sector, sec_t count )
|
|
||||||
{
|
{
|
||||||
|
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
CACHE_ENTRY* cacheEntries = cache->cacheEntries;
|
CACHE_ENTRY* cacheEntries = cache->cacheEntries;
|
||||||
unsigned int numberOfPages = cache->numberOfPages;
|
unsigned int numberOfPages = cache->numberOfPages;
|
||||||
CACHE_ENTRY *entry = NULL;
|
CACHE_ENTRY *entry = NULL;
|
||||||
sec_t lowest = UINT_MAX;
|
sec_t lowest = UINT_MAX;
|
||||||
|
|
||||||
for ( i = 0; i < numberOfPages; i++ )
|
for (i = 0; i < numberOfPages; i++)
|
||||||
{
|
{
|
||||||
if ( cacheEntries[i].sector != CACHE_FREE )
|
if (cacheEntries[i].sector != CACHE_FREE)
|
||||||
{
|
{
|
||||||
bool intersect;
|
bool intersect;
|
||||||
if ( sector > cacheEntries[i].sector )
|
if (sector > cacheEntries[i].sector)
|
||||||
{
|
{
|
||||||
intersect = sector - cacheEntries[i].sector < cacheEntries[i].count;
|
intersect = sector - cacheEntries[i].sector < cacheEntries[i].count;
|
||||||
}
|
}
|
||||||
@ -306,7 +319,7 @@ static CACHE_ENTRY* _FAT_cache_findPage( CACHE *cache, sec_t sector, sec_t count
|
|||||||
intersect = cacheEntries[i].sector - sector < count;
|
intersect = cacheEntries[i].sector - sector < count;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( intersect && ( cacheEntries[i].sector < lowest ) )
|
if (intersect && (cacheEntries[i].sector < lowest))
|
||||||
{
|
{
|
||||||
lowest = cacheEntries[i].sector;
|
lowest = cacheEntries[i].sector;
|
||||||
entry = &cacheEntries[i];
|
entry = &cacheEntries[i];
|
||||||
@ -317,27 +330,27 @@ static CACHE_ENTRY* _FAT_cache_findPage( CACHE *cache, sec_t sector, sec_t count
|
|||||||
return entry;
|
return entry;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool _FAT_cache_writeSectors ( CACHE* cache, sec_t sector, sec_t numSectors, const void* buffer )
|
bool _FAT_cache_writeSectors(CACHE* cache, sec_t sector, sec_t numSectors, const void* buffer)
|
||||||
{
|
{
|
||||||
sec_t sec;
|
sec_t sec;
|
||||||
sec_t secs_to_write;
|
sec_t secs_to_write;
|
||||||
CACHE_ENTRY* entry;
|
CACHE_ENTRY* entry;
|
||||||
const uint8_t *src = buffer;
|
const uint8_t *src = buffer;
|
||||||
|
|
||||||
while ( numSectors > 0 )
|
while (numSectors > 0)
|
||||||
{
|
{
|
||||||
entry = _FAT_cache_findPage( cache, sector, numSectors );
|
entry = _FAT_cache_findPage(cache, sector, numSectors);
|
||||||
|
|
||||||
if ( entry != NULL )
|
if (entry != NULL)
|
||||||
{
|
{
|
||||||
|
|
||||||
if ( entry->sector > sector )
|
if (entry->sector > sector)
|
||||||
{
|
{
|
||||||
|
|
||||||
secs_to_write = entry->sector - sector;
|
secs_to_write = entry->sector - sector;
|
||||||
|
|
||||||
_FAT_disc_writeSectors( cache->disc, sector, secs_to_write, src );
|
_FAT_disc_writeSectors(cache->disc, sector, secs_to_write, src);
|
||||||
src += ( secs_to_write * BYTES_PER_READ );
|
src += (secs_to_write * BYTES_PER_READ);
|
||||||
sector += secs_to_write;
|
sector += secs_to_write;
|
||||||
numSectors -= secs_to_write;
|
numSectors -= secs_to_write;
|
||||||
}
|
}
|
||||||
@ -345,11 +358,11 @@ bool _FAT_cache_writeSectors ( CACHE* cache, sec_t sector, sec_t numSectors, con
|
|||||||
sec = sector - entry->sector;
|
sec = sector - entry->sector;
|
||||||
secs_to_write = entry->count - sec;
|
secs_to_write = entry->count - sec;
|
||||||
|
|
||||||
if ( secs_to_write > numSectors ) secs_to_write = numSectors;
|
if (secs_to_write > numSectors) secs_to_write = numSectors;
|
||||||
|
|
||||||
memcpy( entry->cache + ( sec*BYTES_PER_READ ), src, ( secs_to_write*BYTES_PER_READ ) );
|
memcpy(entry->cache + (sec * BYTES_PER_READ), src, (secs_to_write * BYTES_PER_READ));
|
||||||
|
|
||||||
src += ( secs_to_write * BYTES_PER_READ );
|
src += (secs_to_write * BYTES_PER_READ);
|
||||||
sector += secs_to_write;
|
sector += secs_to_write;
|
||||||
numSectors -= secs_to_write;
|
numSectors -= secs_to_write;
|
||||||
|
|
||||||
@ -358,7 +371,7 @@ bool _FAT_cache_writeSectors ( CACHE* cache, sec_t sector, sec_t numSectors, con
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
_FAT_disc_writeSectors( cache->disc, sector, numSectors, src );
|
_FAT_disc_writeSectors(cache->disc, sector, numSectors, src);
|
||||||
numSectors = 0;
|
numSectors = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -366,17 +379,18 @@ bool _FAT_cache_writeSectors ( CACHE* cache, sec_t sector, sec_t numSectors, con
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Flushes all dirty pages to disc, clearing the dirty flag.
|
Flushes all dirty pages to disc, clearing the dirty flag.
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_flush ( CACHE* cache )
|
bool _FAT_cache_flush(CACHE* cache)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
|
|
||||||
for ( i = 0; i < cache->numberOfPages; i++ )
|
for (i = 0; i < cache->numberOfPages; i++)
|
||||||
{
|
{
|
||||||
if ( cache->cacheEntries[i].dirty )
|
if (cache->cacheEntries[i].dirty)
|
||||||
{
|
{
|
||||||
if ( !_FAT_disc_writeSectors ( cache->disc, cache->cacheEntries[i].sector, cache->cacheEntries[i].count, cache->cacheEntries[i].cache ) )
|
if (!_FAT_disc_writeSectors(cache->disc, cache->cacheEntries[i].sector, cache->cacheEntries[i].count,
|
||||||
|
cache->cacheEntries[i].cache))
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -387,11 +401,11 @@ bool _FAT_cache_flush ( CACHE* cache )
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void _FAT_cache_invalidate ( CACHE* cache )
|
void _FAT_cache_invalidate(CACHE* cache)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
_FAT_cache_flush( cache );
|
_FAT_cache_flush(cache);
|
||||||
for ( i = 0; i < cache->numberOfPages; i++ )
|
for (i = 0; i < cache->numberOfPages; i++)
|
||||||
{
|
{
|
||||||
cache->cacheEntries[i].sector = CACHE_FREE;
|
cache->cacheEntries[i].sector = CACHE_FREE;
|
||||||
cache->cacheEntries[i].last_access = 0;
|
cache->cacheEntries[i].last_access = 0;
|
||||||
|
@ -14,13 +14,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -31,7 +31,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __CACHE_H
|
#ifndef __CACHE_H
|
||||||
#define __CACHE_H
|
#define __CACHE_H
|
||||||
@ -44,91 +44,93 @@
|
|||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
sec_t sector;
|
sec_t sector;
|
||||||
unsigned int count;
|
unsigned int count;
|
||||||
unsigned int last_access;
|
unsigned int last_access;
|
||||||
bool dirty;
|
bool dirty;
|
||||||
uint8_t* cache;
|
uint8_t* cache;
|
||||||
} CACHE_ENTRY;
|
} CACHE_ENTRY;
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
const DISC_INTERFACE* disc;
|
const DISC_INTERFACE* disc;
|
||||||
sec_t endOfPartition;
|
sec_t endOfPartition;
|
||||||
unsigned int numberOfPages;
|
unsigned int numberOfPages;
|
||||||
unsigned int sectorsPerPage;
|
unsigned int sectorsPerPage;
|
||||||
CACHE_ENTRY* cacheEntries;
|
CACHE_ENTRY* cacheEntries;
|
||||||
} CACHE;
|
} CACHE;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Read data from a sector in the cache
|
Read data from a sector in the cache
|
||||||
If the sector is not in the cache, it will be swapped in
|
If the sector is not in the cache, it will be swapped in
|
||||||
offset is the position to start reading from
|
offset is the position to start reading from
|
||||||
size is the amount of data to read
|
size is the amount of data to read
|
||||||
Precondition: offset + size <= BYTES_PER_READ
|
Precondition: offset + size <= BYTES_PER_READ
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_readPartialSector ( CACHE* cache, void* buffer, sec_t sector, unsigned int offset, size_t size );
|
bool _FAT_cache_readPartialSector(CACHE* cache, void* buffer, sec_t sector, unsigned int offset, size_t size);
|
||||||
|
|
||||||
bool _FAT_cache_readLittleEndianValue ( CACHE* cache, uint32_t *value, sec_t sector, unsigned int offset, int num_bytes );
|
bool _FAT_cache_readLittleEndianValue(CACHE* cache, uint32_t *value, sec_t sector, unsigned int offset, int num_bytes);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write data to a sector in the cache
|
Write data to a sector in the cache
|
||||||
If the sector is not in the cache, it will be swapped in.
|
If the sector is not in the cache, it will be swapped in.
|
||||||
When the sector is swapped out, the data will be written to the disc
|
When the sector is swapped out, the data will be written to the disc
|
||||||
offset is the position to start writing to
|
offset is the position to start writing to
|
||||||
size is the amount of data to write
|
size is the amount of data to write
|
||||||
Precondition: offset + size <= BYTES_PER_READ
|
Precondition: offset + size <= BYTES_PER_READ
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_writePartialSector ( CACHE* cache, const void* buffer, sec_t sector, unsigned int offset, size_t size );
|
bool _FAT_cache_writePartialSector(CACHE* cache, const void* buffer, sec_t sector, unsigned int offset, size_t size);
|
||||||
|
|
||||||
bool _FAT_cache_writeLittleEndianValue ( CACHE* cache, const uint32_t value, sec_t sector, unsigned int offset, int num_bytes );
|
bool _FAT_cache_writeLittleEndianValue(CACHE* cache, const uint32_t value, sec_t sector, unsigned int offset,
|
||||||
|
int num_bytes);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write data to a sector in the cache, zeroing the sector first
|
Write data to a sector in the cache, zeroing the sector first
|
||||||
If the sector is not in the cache, it will be swapped in.
|
If the sector is not in the cache, it will be swapped in.
|
||||||
When the sector is swapped out, the data will be written to the disc
|
When the sector is swapped out, the data will be written to the disc
|
||||||
offset is the position to start writing to
|
offset is the position to start writing to
|
||||||
size is the amount of data to write
|
size is the amount of data to write
|
||||||
Precondition: offset + size <= BYTES_PER_READ
|
Precondition: offset + size <= BYTES_PER_READ
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_eraseWritePartialSector ( CACHE* cache, const void* buffer, sec_t sector, unsigned int offset, size_t size );
|
bool _FAT_cache_eraseWritePartialSector(CACHE* cache, const void* buffer, sec_t sector, unsigned int offset,
|
||||||
|
size_t size);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Read several sectors from the cache
|
Read several sectors from the cache
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_readSectors ( CACHE* cache, sec_t sector, sec_t numSectors, void* buffer );
|
bool _FAT_cache_readSectors(CACHE* cache, sec_t sector, sec_t numSectors, void* buffer);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Read a full sector from the cache
|
Read a full sector from the cache
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_cache_readSector ( CACHE* cache, void* buffer, sec_t sector )
|
static inline bool _FAT_cache_readSector(CACHE* cache, void* buffer, sec_t sector)
|
||||||
{
|
{
|
||||||
return _FAT_cache_readPartialSector ( cache, buffer, sector, 0, BYTES_PER_READ );
|
return _FAT_cache_readPartialSector(cache, buffer, sector, 0, BYTES_PER_READ);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write a full sector to the cache
|
Write a full sector to the cache
|
||||||
*/
|
*/
|
||||||
static inline bool _FAT_cache_writeSector ( CACHE* cache, const void* buffer, sec_t sector )
|
static inline bool _FAT_cache_writeSector(CACHE* cache, const void* buffer, sec_t sector)
|
||||||
{
|
{
|
||||||
return _FAT_cache_writePartialSector ( cache, buffer, sector, 0, BYTES_PER_READ );
|
return _FAT_cache_writePartialSector(cache, buffer, sector, 0, BYTES_PER_READ);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool _FAT_cache_writeSectors ( CACHE* cache, sec_t sector, sec_t numSectors, const void* buffer );
|
bool _FAT_cache_writeSectors(CACHE* cache, sec_t sector, sec_t numSectors, const void* buffer);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write any dirty sectors back to disc and clear out the contents of the cache
|
Write any dirty sectors back to disc and clear out the contents of the cache
|
||||||
*/
|
*/
|
||||||
bool _FAT_cache_flush ( CACHE* cache );
|
bool _FAT_cache_flush(CACHE* cache);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Clear out the contents of the cache without writing any dirty sectors first
|
Clear out the contents of the cache without writing any dirty sectors first
|
||||||
*/
|
*/
|
||||||
void _FAT_cache_invalidate ( CACHE* cache );
|
void _FAT_cache_invalidate(CACHE* cache);
|
||||||
|
|
||||||
CACHE* _FAT_cache_constructor ( unsigned int numberOfPages, unsigned int sectorsPerPage, const DISC_INTERFACE* discInterface, sec_t endOfPartition );
|
CACHE* _FAT_cache_constructor(unsigned int numberOfPages, unsigned int sectorsPerPage,
|
||||||
|
const DISC_INTERFACE* discInterface, sec_t endOfPartition);
|
||||||
|
|
||||||
void _FAT_cache_destructor ( CACHE* cache );
|
void _FAT_cache_destructor(CACHE* cache);
|
||||||
|
|
||||||
#endif // _CACHE_H
|
#endif // _CACHE_H
|
||||||
|
|
||||||
|
@ -9,13 +9,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -26,7 +26,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
@ -44,55 +44,54 @@
|
|||||||
#include "filetime.h"
|
#include "filetime.h"
|
||||||
#include "lock.h"
|
#include "lock.h"
|
||||||
|
|
||||||
|
int _FAT_stat_r(struct _reent *r, const char *path, struct stat *st)
|
||||||
int _FAT_stat_r ( struct _reent *r, const char *path, struct stat *st )
|
|
||||||
{
|
{
|
||||||
PARTITION* partition = NULL;
|
PARTITION* partition = NULL;
|
||||||
DIR_ENTRY dirEntry;
|
DIR_ENTRY dirEntry;
|
||||||
|
|
||||||
// Get the partition this file is on
|
// Get the partition this file is on
|
||||||
partition = _FAT_partition_getPartitionFromPath ( path );
|
partition = _FAT_partition_getPartitionFromPath(path);
|
||||||
if ( partition == NULL )
|
if (partition == NULL)
|
||||||
{
|
{
|
||||||
r->_errno = ENODEV;
|
r->_errno = ENODEV;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Move the path pointer to the start of the actual path
|
// Move the path pointer to the start of the actual path
|
||||||
if ( strchr ( path, ':' ) != NULL )
|
if (strchr(path, ':') != NULL)
|
||||||
{
|
{
|
||||||
path = strchr ( path, ':' ) + 1;
|
path = strchr(path, ':') + 1;
|
||||||
}
|
}
|
||||||
if ( strchr ( path, ':' ) != NULL )
|
if (strchr(path, ':') != NULL)
|
||||||
{
|
{
|
||||||
r->_errno = EINVAL;
|
r->_errno = EINVAL;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
_FAT_lock( &partition->lock );
|
_FAT_lock(&partition->lock);
|
||||||
|
|
||||||
// Search for the file on the disc
|
// Search for the file on the disc
|
||||||
if ( !_FAT_directory_entryFromPath ( partition, &dirEntry, path, NULL ) )
|
if (!_FAT_directory_entryFromPath(partition, &dirEntry, path, NULL))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = ENOENT;
|
r->_errno = ENOENT;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fill in the stat struct
|
// Fill in the stat struct
|
||||||
_FAT_directory_entryStat ( partition, &dirEntry, st );
|
_FAT_directory_entryStat(partition, &dirEntry, st);
|
||||||
|
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int _FAT_link_r ( struct _reent *r, const char *existing, const char *newLink )
|
int _FAT_link_r(struct _reent *r, const char *existing, const char *newLink)
|
||||||
{
|
{
|
||||||
r->_errno = ENOTSUP;
|
r->_errno = ENOTSUP;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int _FAT_unlink_r ( struct _reent *r, const char *path )
|
int _FAT_unlink_r(struct _reent *r, const char *path)
|
||||||
{
|
{
|
||||||
PARTITION* partition = NULL;
|
PARTITION* partition = NULL;
|
||||||
DIR_ENTRY dirEntry;
|
DIR_ENTRY dirEntry;
|
||||||
@ -102,66 +101,65 @@ int _FAT_unlink_r ( struct _reent *r, const char *path )
|
|||||||
bool errorOccured = false;
|
bool errorOccured = false;
|
||||||
|
|
||||||
// Get the partition this directory is on
|
// Get the partition this directory is on
|
||||||
partition = _FAT_partition_getPartitionFromPath ( path );
|
partition = _FAT_partition_getPartitionFromPath(path);
|
||||||
if ( partition == NULL )
|
if (partition == NULL)
|
||||||
{
|
{
|
||||||
r->_errno = ENODEV;
|
r->_errno = ENODEV;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make sure we aren't trying to write to a read-only disc
|
// Make sure we aren't trying to write to a read-only disc
|
||||||
if ( partition->readOnly )
|
if (partition->readOnly)
|
||||||
{
|
{
|
||||||
r->_errno = EROFS;
|
r->_errno = EROFS;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Move the path pointer to the start of the actual path
|
// Move the path pointer to the start of the actual path
|
||||||
if ( strchr ( path, ':' ) != NULL )
|
if (strchr(path, ':') != NULL)
|
||||||
{
|
{
|
||||||
path = strchr ( path, ':' ) + 1;
|
path = strchr(path, ':') + 1;
|
||||||
}
|
}
|
||||||
if ( strchr ( path, ':' ) != NULL )
|
if (strchr(path, ':') != NULL)
|
||||||
{
|
{
|
||||||
r->_errno = EINVAL;
|
r->_errno = EINVAL;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
_FAT_lock( &partition->lock );
|
_FAT_lock(&partition->lock);
|
||||||
|
|
||||||
// Search for the file on the disc
|
// Search for the file on the disc
|
||||||
if ( !_FAT_directory_entryFromPath ( partition, &dirEntry, path, NULL ) )
|
if (!_FAT_directory_entryFromPath(partition, &dirEntry, path, NULL))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = ENOENT;
|
r->_errno = ENOENT;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
cluster = _FAT_directory_entryGetCluster ( partition, dirEntry.entryData );
|
cluster = _FAT_directory_entryGetCluster(partition, dirEntry.entryData);
|
||||||
|
|
||||||
|
|
||||||
// If this is a directory, make sure it is empty
|
// If this is a directory, make sure it is empty
|
||||||
if ( _FAT_directory_isDirectory ( &dirEntry ) )
|
if (_FAT_directory_isDirectory(&dirEntry))
|
||||||
{
|
{
|
||||||
nextEntry = _FAT_directory_getFirstEntry ( partition, &dirContents, cluster );
|
nextEntry = _FAT_directory_getFirstEntry(partition, &dirContents, cluster);
|
||||||
|
|
||||||
while ( nextEntry )
|
while (nextEntry)
|
||||||
{
|
{
|
||||||
if ( !_FAT_directory_isDot ( &dirContents ) )
|
if (!_FAT_directory_isDot(&dirContents))
|
||||||
{
|
{
|
||||||
// The directory had something in it that isn't a reference to itself or it's parent
|
// The directory had something in it that isn't a reference to itself or it's parent
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = EPERM;
|
r->_errno = EPERM;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
nextEntry = _FAT_directory_getNextEntry ( partition, &dirContents );
|
nextEntry = _FAT_directory_getNextEntry(partition, &dirContents);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( _FAT_fat_isValidCluster( partition, cluster ) )
|
if (_FAT_fat_isValidCluster(partition, cluster))
|
||||||
{
|
{
|
||||||
// Remove the cluster chain for this file
|
// Remove the cluster chain for this file
|
||||||
if ( !_FAT_fat_clearLinks ( partition, cluster ) )
|
if (!_FAT_fat_clearLinks(partition, cluster))
|
||||||
{
|
{
|
||||||
r->_errno = EIO;
|
r->_errno = EIO;
|
||||||
errorOccured = true;
|
errorOccured = true;
|
||||||
@ -169,21 +167,21 @@ int _FAT_unlink_r ( struct _reent *r, const char *path )
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Remove the directory entry for this file
|
// Remove the directory entry for this file
|
||||||
if ( !_FAT_directory_removeEntry ( partition, &dirEntry ) )
|
if (!_FAT_directory_removeEntry(partition, &dirEntry))
|
||||||
{
|
{
|
||||||
r->_errno = EIO;
|
r->_errno = EIO;
|
||||||
errorOccured = true;
|
errorOccured = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Flush any sectors in the disc cache
|
// Flush any sectors in the disc cache
|
||||||
if ( !_FAT_cache_flush( partition->cache ) )
|
if (!_FAT_cache_flush(partition->cache))
|
||||||
{
|
{
|
||||||
r->_errno = EIO;
|
r->_errno = EIO;
|
||||||
errorOccured = true;
|
errorOccured = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
if ( errorOccured )
|
if (errorOccured)
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@ -193,48 +191,48 @@ int _FAT_unlink_r ( struct _reent *r, const char *path )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int _FAT_chdir_r ( struct _reent *r, const char *path )
|
int _FAT_chdir_r(struct _reent *r, const char *path)
|
||||||
{
|
{
|
||||||
PARTITION* partition = NULL;
|
PARTITION* partition = NULL;
|
||||||
|
|
||||||
// Get the partition this directory is on
|
// Get the partition this directory is on
|
||||||
partition = _FAT_partition_getPartitionFromPath ( path );
|
partition = _FAT_partition_getPartitionFromPath(path);
|
||||||
if ( partition == NULL )
|
if (partition == NULL)
|
||||||
{
|
{
|
||||||
r->_errno = ENODEV;
|
r->_errno = ENODEV;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Move the path pointer to the start of the actual path
|
// Move the path pointer to the start of the actual path
|
||||||
if ( strchr ( path, ':' ) != NULL )
|
if (strchr(path, ':') != NULL)
|
||||||
{
|
{
|
||||||
path = strchr ( path, ':' ) + 1;
|
path = strchr(path, ':') + 1;
|
||||||
}
|
}
|
||||||
if ( strchr ( path, ':' ) != NULL )
|
if (strchr(path, ':') != NULL)
|
||||||
{
|
{
|
||||||
r->_errno = EINVAL;
|
r->_errno = EINVAL;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
_FAT_lock( &partition->lock );
|
_FAT_lock(&partition->lock);
|
||||||
|
|
||||||
// Try changing directory
|
// Try changing directory
|
||||||
if ( _FAT_directory_chdir ( partition, path ) )
|
if (_FAT_directory_chdir(partition, path))
|
||||||
{
|
{
|
||||||
// Successful
|
// Successful
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Failed
|
// Failed
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = ENOTDIR;
|
r->_errno = ENOTDIR;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int _FAT_rename_r ( struct _reent *r, const char *oldName, const char *newName )
|
int _FAT_rename_r(struct _reent *r, const char *oldName, const char *newName)
|
||||||
{
|
{
|
||||||
PARTITION* partition = NULL;
|
PARTITION* partition = NULL;
|
||||||
DIR_ENTRY oldDirEntry;
|
DIR_ENTRY oldDirEntry;
|
||||||
@ -243,73 +241,73 @@ int _FAT_rename_r ( struct _reent *r, const char *oldName, const char *newName )
|
|||||||
uint32_t dirCluster;
|
uint32_t dirCluster;
|
||||||
|
|
||||||
// Get the partition this directory is on
|
// Get the partition this directory is on
|
||||||
partition = _FAT_partition_getPartitionFromPath ( oldName );
|
partition = _FAT_partition_getPartitionFromPath(oldName);
|
||||||
if ( partition == NULL )
|
if (partition == NULL)
|
||||||
{
|
{
|
||||||
r->_errno = ENODEV;
|
r->_errno = ENODEV;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
_FAT_lock( &partition->lock );
|
_FAT_lock(&partition->lock);
|
||||||
|
|
||||||
// Make sure the same partition is used for the old and new names
|
// Make sure the same partition is used for the old and new names
|
||||||
if ( partition != _FAT_partition_getPartitionFromPath ( newName ) )
|
if (partition != _FAT_partition_getPartitionFromPath(newName))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = EXDEV;
|
r->_errno = EXDEV;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make sure we aren't trying to write to a read-only disc
|
// Make sure we aren't trying to write to a read-only disc
|
||||||
if ( partition->readOnly )
|
if (partition->readOnly)
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = EROFS;
|
r->_errno = EROFS;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Move the path pointer to the start of the actual path
|
// Move the path pointer to the start of the actual path
|
||||||
if ( strchr ( oldName, ':' ) != NULL )
|
if (strchr(oldName, ':') != NULL)
|
||||||
{
|
{
|
||||||
oldName = strchr ( oldName, ':' ) + 1;
|
oldName = strchr(oldName, ':') + 1;
|
||||||
}
|
}
|
||||||
if ( strchr ( oldName, ':' ) != NULL )
|
if (strchr(oldName, ':') != NULL)
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = EINVAL;
|
r->_errno = EINVAL;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
if ( strchr ( newName, ':' ) != NULL )
|
if (strchr(newName, ':') != NULL)
|
||||||
{
|
{
|
||||||
newName = strchr ( newName, ':' ) + 1;
|
newName = strchr(newName, ':') + 1;
|
||||||
}
|
}
|
||||||
if ( strchr ( newName, ':' ) != NULL )
|
if (strchr(newName, ':') != NULL)
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = EINVAL;
|
r->_errno = EINVAL;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Search for the file on the disc
|
// Search for the file on the disc
|
||||||
if ( !_FAT_directory_entryFromPath ( partition, &oldDirEntry, oldName, NULL ) )
|
if (!_FAT_directory_entryFromPath(partition, &oldDirEntry, oldName, NULL))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = ENOENT;
|
r->_errno = ENOENT;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make sure there is no existing file / directory with the new name
|
// Make sure there is no existing file / directory with the new name
|
||||||
if ( _FAT_directory_entryFromPath ( partition, &newDirEntry, newName, NULL ) )
|
if (_FAT_directory_entryFromPath(partition, &newDirEntry, newName, NULL))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = EEXIST;
|
r->_errno = EEXIST;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create the new file entry
|
// Create the new file entry
|
||||||
// Get the directory it has to go in
|
// Get the directory it has to go in
|
||||||
pathEnd = strrchr ( newName, DIR_SEPARATOR );
|
pathEnd = strrchr(newName, DIR_SEPARATOR);
|
||||||
if ( pathEnd == NULL )
|
if (pathEnd == NULL)
|
||||||
{
|
{
|
||||||
// No path was specified
|
// No path was specified
|
||||||
dirCluster = partition->cwdCluster;
|
dirCluster = partition->cwdCluster;
|
||||||
@ -319,53 +317,53 @@ int _FAT_rename_r ( struct _reent *r, const char *oldName, const char *newName )
|
|||||||
{
|
{
|
||||||
// Path was specified -- get the right dirCluster
|
// Path was specified -- get the right dirCluster
|
||||||
// Recycling newDirEntry, since it needs to be recreated anyway
|
// Recycling newDirEntry, since it needs to be recreated anyway
|
||||||
if ( !_FAT_directory_entryFromPath ( partition, &newDirEntry, newName, pathEnd ) ||
|
if (!_FAT_directory_entryFromPath(partition, &newDirEntry, newName, pathEnd) || !_FAT_directory_isDirectory(
|
||||||
!_FAT_directory_isDirectory( &newDirEntry ) )
|
&newDirEntry))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = ENOTDIR;
|
r->_errno = ENOTDIR;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
dirCluster = _FAT_directory_entryGetCluster ( partition, newDirEntry.entryData );
|
dirCluster = _FAT_directory_entryGetCluster(partition, newDirEntry.entryData);
|
||||||
// Move the pathEnd past the last DIR_SEPARATOR
|
// Move the pathEnd past the last DIR_SEPARATOR
|
||||||
pathEnd += 1;
|
pathEnd += 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy the entry data
|
// Copy the entry data
|
||||||
memcpy ( &newDirEntry, &oldDirEntry, sizeof( DIR_ENTRY ) );
|
memcpy(&newDirEntry, &oldDirEntry, sizeof(DIR_ENTRY));
|
||||||
|
|
||||||
// Set the new name
|
// Set the new name
|
||||||
strncpy ( newDirEntry.filename, pathEnd, MAX_FILENAME_LENGTH - 1 );
|
strncpy(newDirEntry.filename, pathEnd, MAX_FILENAME_LENGTH - 1);
|
||||||
|
|
||||||
// Write the new entry
|
// Write the new entry
|
||||||
if ( !_FAT_directory_addEntry ( partition, &newDirEntry, dirCluster ) )
|
if (!_FAT_directory_addEntry(partition, &newDirEntry, dirCluster))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = ENOSPC;
|
r->_errno = ENOSPC;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove the old entry
|
// Remove the old entry
|
||||||
if ( !_FAT_directory_removeEntry ( partition, &oldDirEntry ) )
|
if (!_FAT_directory_removeEntry(partition, &oldDirEntry))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = EIO;
|
r->_errno = EIO;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Flush any sectors in the disc cache
|
// Flush any sectors in the disc cache
|
||||||
if ( !_FAT_cache_flush ( partition->cache ) )
|
if (!_FAT_cache_flush(partition->cache))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = EIO;
|
r->_errno = EIO;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int _FAT_mkdir_r ( struct _reent *r, const char *path, int mode )
|
int _FAT_mkdir_r(struct _reent *r, const char *path, int mode)
|
||||||
{
|
{
|
||||||
PARTITION* partition = NULL;
|
PARTITION* partition = NULL;
|
||||||
bool fileExists;
|
bool fileExists;
|
||||||
@ -374,48 +372,48 @@ int _FAT_mkdir_r ( struct _reent *r, const char *path, int mode )
|
|||||||
uint32_t parentCluster, dirCluster;
|
uint32_t parentCluster, dirCluster;
|
||||||
uint8_t newEntryData[DIR_ENTRY_DATA_SIZE];
|
uint8_t newEntryData[DIR_ENTRY_DATA_SIZE];
|
||||||
|
|
||||||
partition = _FAT_partition_getPartitionFromPath ( path );
|
partition = _FAT_partition_getPartitionFromPath(path);
|
||||||
if ( partition == NULL )
|
if (partition == NULL)
|
||||||
{
|
{
|
||||||
r->_errno = ENODEV;
|
r->_errno = ENODEV;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Move the path pointer to the start of the actual path
|
// Move the path pointer to the start of the actual path
|
||||||
if ( strchr ( path, ':' ) != NULL )
|
if (strchr(path, ':') != NULL)
|
||||||
{
|
{
|
||||||
path = strchr ( path, ':' ) + 1;
|
path = strchr(path, ':') + 1;
|
||||||
}
|
}
|
||||||
if ( strchr ( path, ':' ) != NULL )
|
if (strchr(path, ':') != NULL)
|
||||||
{
|
{
|
||||||
r->_errno = EINVAL;
|
r->_errno = EINVAL;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
_FAT_lock( &partition->lock );
|
_FAT_lock(&partition->lock);
|
||||||
|
|
||||||
// Search for the file/directory on the disc
|
// Search for the file/directory on the disc
|
||||||
fileExists = _FAT_directory_entryFromPath ( partition, &dirEntry, path, NULL );
|
fileExists = _FAT_directory_entryFromPath(partition, &dirEntry, path, NULL);
|
||||||
|
|
||||||
// Make sure it doesn't exist
|
// Make sure it doesn't exist
|
||||||
if ( fileExists )
|
if (fileExists)
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = EEXIST;
|
r->_errno = EEXIST;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( partition->readOnly )
|
if (partition->readOnly)
|
||||||
{
|
{
|
||||||
// We can't write to a read-only partition
|
// We can't write to a read-only partition
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = EROFS;
|
r->_errno = EROFS;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the directory it has to go in
|
// Get the directory it has to go in
|
||||||
pathEnd = strrchr ( path, DIR_SEPARATOR );
|
pathEnd = strrchr(path, DIR_SEPARATOR);
|
||||||
if ( pathEnd == NULL )
|
if (pathEnd == NULL)
|
||||||
{
|
{
|
||||||
// No path was specified
|
// No path was specified
|
||||||
parentCluster = partition->cwdCluster;
|
parentCluster = partition->cwdCluster;
|
||||||
@ -425,259 +423,254 @@ int _FAT_mkdir_r ( struct _reent *r, const char *path, int mode )
|
|||||||
{
|
{
|
||||||
// Path was specified -- get the right parentCluster
|
// Path was specified -- get the right parentCluster
|
||||||
// Recycling dirEntry, since it needs to be recreated anyway
|
// Recycling dirEntry, since it needs to be recreated anyway
|
||||||
if ( !_FAT_directory_entryFromPath ( partition, &dirEntry, path, pathEnd ) ||
|
if (!_FAT_directory_entryFromPath(partition, &dirEntry, path, pathEnd)
|
||||||
!_FAT_directory_isDirectory( &dirEntry ) )
|
|| !_FAT_directory_isDirectory(&dirEntry))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = ENOTDIR;
|
r->_errno = ENOTDIR;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
parentCluster = _FAT_directory_entryGetCluster ( partition, dirEntry.entryData );
|
parentCluster = _FAT_directory_entryGetCluster(partition, dirEntry.entryData);
|
||||||
// Move the pathEnd past the last DIR_SEPARATOR
|
// Move the pathEnd past the last DIR_SEPARATOR
|
||||||
pathEnd += 1;
|
pathEnd += 1;
|
||||||
}
|
}
|
||||||
// Create the entry data
|
// Create the entry data
|
||||||
strncpy ( dirEntry.filename, pathEnd, MAX_FILENAME_LENGTH - 1 );
|
strncpy(dirEntry.filename, pathEnd, MAX_FILENAME_LENGTH - 1);
|
||||||
memset ( dirEntry.entryData, 0, DIR_ENTRY_DATA_SIZE );
|
memset(dirEntry.entryData, 0, DIR_ENTRY_DATA_SIZE);
|
||||||
|
|
||||||
// Set the creation time and date
|
// Set the creation time and date
|
||||||
dirEntry.entryData[DIR_ENTRY_cTime_ms] = 0;
|
dirEntry.entryData[DIR_ENTRY_cTime_ms] = 0;
|
||||||
u16_to_u8array ( dirEntry.entryData, DIR_ENTRY_cTime, _FAT_filetime_getTimeFromRTC() );
|
u16_to_u8array(dirEntry.entryData, DIR_ENTRY_cTime, _FAT_filetime_getTimeFromRTC());
|
||||||
u16_to_u8array ( dirEntry.entryData, DIR_ENTRY_cDate, _FAT_filetime_getDateFromRTC() );
|
u16_to_u8array(dirEntry.entryData, DIR_ENTRY_cDate, _FAT_filetime_getDateFromRTC());
|
||||||
u16_to_u8array ( dirEntry.entryData, DIR_ENTRY_mTime, _FAT_filetime_getTimeFromRTC() );
|
u16_to_u8array(dirEntry.entryData, DIR_ENTRY_mTime, _FAT_filetime_getTimeFromRTC());
|
||||||
u16_to_u8array ( dirEntry.entryData, DIR_ENTRY_mDate, _FAT_filetime_getDateFromRTC() );
|
u16_to_u8array(dirEntry.entryData, DIR_ENTRY_mDate, _FAT_filetime_getDateFromRTC());
|
||||||
u16_to_u8array ( dirEntry.entryData, DIR_ENTRY_aDate, _FAT_filetime_getDateFromRTC() );
|
u16_to_u8array(dirEntry.entryData, DIR_ENTRY_aDate, _FAT_filetime_getDateFromRTC());
|
||||||
|
|
||||||
// Set the directory attribute
|
// Set the directory attribute
|
||||||
dirEntry.entryData[DIR_ENTRY_attributes] = ATTRIB_DIR;
|
dirEntry.entryData[DIR_ENTRY_attributes] = ATTRIB_DIR;
|
||||||
|
|
||||||
// Get a cluster for the new directory
|
// Get a cluster for the new directory
|
||||||
dirCluster = _FAT_fat_linkFreeClusterCleared ( partition, CLUSTER_FREE );
|
dirCluster = _FAT_fat_linkFreeClusterCleared(partition, CLUSTER_FREE);
|
||||||
if ( !_FAT_fat_isValidCluster( partition, dirCluster ) )
|
if (!_FAT_fat_isValidCluster(partition, dirCluster))
|
||||||
{
|
{
|
||||||
// No space left on disc for the cluster
|
// No space left on disc for the cluster
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = ENOSPC;
|
r->_errno = ENOSPC;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
u16_to_u8array ( dirEntry.entryData, DIR_ENTRY_cluster, dirCluster );
|
u16_to_u8array(dirEntry.entryData, DIR_ENTRY_cluster, dirCluster);
|
||||||
u16_to_u8array ( dirEntry.entryData, DIR_ENTRY_clusterHigh, dirCluster >> 16 );
|
u16_to_u8array(dirEntry.entryData, DIR_ENTRY_clusterHigh, dirCluster >> 16);
|
||||||
|
|
||||||
// Write the new directory's entry to it's parent
|
// Write the new directory's entry to it's parent
|
||||||
if ( !_FAT_directory_addEntry ( partition, &dirEntry, parentCluster ) )
|
if (!_FAT_directory_addEntry(partition, &dirEntry, parentCluster))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = ENOSPC;
|
r->_errno = ENOSPC;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create the dot entry within the directory
|
// Create the dot entry within the directory
|
||||||
memset ( newEntryData, 0, DIR_ENTRY_DATA_SIZE );
|
memset(newEntryData, 0, DIR_ENTRY_DATA_SIZE);
|
||||||
memset ( newEntryData, ' ', 11 );
|
memset(newEntryData, ' ', 11);
|
||||||
newEntryData[DIR_ENTRY_name] = '.';
|
newEntryData[DIR_ENTRY_name] = '.';
|
||||||
newEntryData[DIR_ENTRY_attributes] = ATTRIB_DIR;
|
newEntryData[DIR_ENTRY_attributes] = ATTRIB_DIR;
|
||||||
u16_to_u8array ( newEntryData, DIR_ENTRY_cluster, dirCluster );
|
u16_to_u8array(newEntryData, DIR_ENTRY_cluster, dirCluster);
|
||||||
u16_to_u8array ( newEntryData, DIR_ENTRY_clusterHigh, dirCluster >> 16 );
|
u16_to_u8array(newEntryData, DIR_ENTRY_clusterHigh, dirCluster >> 16);
|
||||||
|
|
||||||
// Write it to the directory, erasing that sector in the process
|
// Write it to the directory, erasing that sector in the process
|
||||||
_FAT_cache_eraseWritePartialSector ( partition->cache, newEntryData,
|
_FAT_cache_eraseWritePartialSector(partition->cache, newEntryData, _FAT_fat_clusterToSector(partition, dirCluster),
|
||||||
_FAT_fat_clusterToSector ( partition, dirCluster ), 0, DIR_ENTRY_DATA_SIZE );
|
0, DIR_ENTRY_DATA_SIZE);
|
||||||
|
|
||||||
|
|
||||||
// Create the double dot entry within the directory
|
// Create the double dot entry within the directory
|
||||||
|
|
||||||
// if ParentDir == Rootdir then ".."" always link to Cluster 0
|
// if ParentDir == Rootdir then ".."" always link to Cluster 0
|
||||||
if ( parentCluster == partition->rootDirCluster )
|
if (parentCluster == partition->rootDirCluster) parentCluster = FAT16_ROOT_DIR_CLUSTER;
|
||||||
parentCluster = FAT16_ROOT_DIR_CLUSTER;
|
|
||||||
|
|
||||||
newEntryData[DIR_ENTRY_name + 1] = '.';
|
newEntryData[DIR_ENTRY_name + 1] = '.';
|
||||||
u16_to_u8array ( newEntryData, DIR_ENTRY_cluster, parentCluster );
|
u16_to_u8array(newEntryData, DIR_ENTRY_cluster, parentCluster);
|
||||||
u16_to_u8array ( newEntryData, DIR_ENTRY_clusterHigh, parentCluster >> 16 );
|
u16_to_u8array(newEntryData, DIR_ENTRY_clusterHigh, parentCluster >> 16);
|
||||||
|
|
||||||
// Write it to the directory
|
// Write it to the directory
|
||||||
_FAT_cache_writePartialSector ( partition->cache, newEntryData,
|
_FAT_cache_writePartialSector(partition->cache, newEntryData, _FAT_fat_clusterToSector(partition, dirCluster),
|
||||||
_FAT_fat_clusterToSector ( partition, dirCluster ), DIR_ENTRY_DATA_SIZE, DIR_ENTRY_DATA_SIZE );
|
DIR_ENTRY_DATA_SIZE, DIR_ENTRY_DATA_SIZE);
|
||||||
|
|
||||||
// Flush any sectors in the disc cache
|
// Flush any sectors in the disc cache
|
||||||
if ( !_FAT_cache_flush( partition->cache ) )
|
if (!_FAT_cache_flush(partition->cache))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
r->_errno = EIO;
|
r->_errno = EIO;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int _FAT_statvfs_r ( struct _reent *r, const char *path, struct statvfs *buf )
|
int _FAT_statvfs_r(struct _reent *r, const char *path, struct statvfs *buf)
|
||||||
{
|
{
|
||||||
PARTITION* partition = NULL;
|
PARTITION* partition = NULL;
|
||||||
unsigned int freeClusterCount;
|
unsigned int freeClusterCount;
|
||||||
|
|
||||||
// Get the partition of the requested path
|
// Get the partition of the requested path
|
||||||
partition = _FAT_partition_getPartitionFromPath ( path );
|
partition = _FAT_partition_getPartitionFromPath(path);
|
||||||
if ( partition == NULL )
|
if (partition == NULL)
|
||||||
{
|
{
|
||||||
r->_errno = ENODEV;
|
r->_errno = ENODEV;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
_FAT_lock( &partition->lock );
|
_FAT_lock(&partition->lock);
|
||||||
|
|
||||||
freeClusterCount = _FAT_fat_freeClusterCount ( partition );
|
freeClusterCount = _FAT_fat_freeClusterCount(partition);
|
||||||
|
|
||||||
// FAT clusters = POSIX blocks
|
// FAT clusters = POSIX blocks
|
||||||
buf->f_bsize = partition->bytesPerCluster; // File system block size.
|
buf->f_bsize = partition->bytesPerCluster; // File system block size.
|
||||||
buf->f_frsize = partition->bytesPerCluster; // Fundamental file system block size.
|
buf->f_frsize = partition->bytesPerCluster; // Fundamental file system block size.
|
||||||
|
|
||||||
buf->f_blocks = partition->fat.lastCluster - CLUSTER_FIRST + 1; // Total number of blocks on file system in units of f_frsize.
|
buf->f_blocks = partition->fat.lastCluster - CLUSTER_FIRST + 1; // Total number of blocks on file system in units of f_frsize.
|
||||||
buf->f_bfree = freeClusterCount; // Total number of free blocks.
|
buf->f_bfree = freeClusterCount; // Total number of free blocks.
|
||||||
buf->f_bavail = freeClusterCount; // Number of free blocks available to non-privileged process.
|
buf->f_bavail = freeClusterCount; // Number of free blocks available to non-privileged process.
|
||||||
|
|
||||||
// Treat requests for info on inodes as clusters
|
// Treat requests for info on inodes as clusters
|
||||||
buf->f_files = partition->fat.lastCluster - CLUSTER_FIRST + 1; // Total number of file serial numbers.
|
buf->f_files = partition->fat.lastCluster - CLUSTER_FIRST + 1; // Total number of file serial numbers.
|
||||||
buf->f_ffree = freeClusterCount; // Total number of free file serial numbers.
|
buf->f_ffree = freeClusterCount; // Total number of free file serial numbers.
|
||||||
buf->f_favail = freeClusterCount; // Number of file serial numbers available to non-privileged process.
|
buf->f_favail = freeClusterCount; // Number of file serial numbers available to non-privileged process.
|
||||||
|
|
||||||
// File system ID. 32bit ioType value
|
// File system ID. 32bit ioType value
|
||||||
buf->f_fsid = _FAT_disc_hostType( partition->disc );
|
buf->f_fsid = _FAT_disc_hostType(partition->disc);
|
||||||
|
|
||||||
// Bit mask of f_flag values.
|
// Bit mask of f_flag values.
|
||||||
buf->f_flag = ST_NOSUID /* No support for ST_ISUID and ST_ISGID file mode bits */
|
buf->f_flag = ST_NOSUID /* No support for ST_ISUID and ST_ISGID file mode bits */
|
||||||
| ( partition->readOnly ? ST_RDONLY /* Read only file system */ : 0 ) ;
|
| (partition->readOnly ? ST_RDONLY /* Read only file system */: 0);
|
||||||
// Maximum filename length.
|
// Maximum filename length.
|
||||||
buf->f_namemax = MAX_FILENAME_LENGTH;
|
buf->f_namemax = MAX_FILENAME_LENGTH;
|
||||||
|
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
DIR_ITER* _FAT_diropen_r( struct _reent *r, DIR_ITER *dirState, const char *path )
|
DIR_ITER* _FAT_diropen_r(struct _reent *r, DIR_ITER *dirState, const char *path)
|
||||||
{
|
{
|
||||||
DIR_ENTRY dirEntry;
|
DIR_ENTRY dirEntry;
|
||||||
DIR_STATE_STRUCT* state = ( DIR_STATE_STRUCT* ) ( dirState->dirStruct );
|
DIR_STATE_STRUCT* state = (DIR_STATE_STRUCT*) (dirState->dirStruct);
|
||||||
bool fileExists;
|
bool fileExists;
|
||||||
|
|
||||||
state->partition = _FAT_partition_getPartitionFromPath ( path );
|
state->partition = _FAT_partition_getPartitionFromPath(path);
|
||||||
if ( state->partition == NULL )
|
if (state->partition == NULL)
|
||||||
{
|
{
|
||||||
r->_errno = ENODEV;
|
r->_errno = ENODEV;
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Move the path pointer to the start of the actual path
|
// Move the path pointer to the start of the actual path
|
||||||
if ( strchr ( path, ':' ) != NULL )
|
if (strchr(path, ':') != NULL)
|
||||||
{
|
{
|
||||||
path = strchr ( path, ':' ) + 1;
|
path = strchr(path, ':') + 1;
|
||||||
}
|
}
|
||||||
if ( strchr ( path, ':' ) != NULL )
|
if (strchr(path, ':') != NULL)
|
||||||
{
|
{
|
||||||
r->_errno = EINVAL;
|
r->_errno = EINVAL;
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
_FAT_lock( &state->partition->lock );
|
_FAT_lock(&state->partition->lock);
|
||||||
|
|
||||||
// Get the start cluster of the directory
|
// Get the start cluster of the directory
|
||||||
fileExists = _FAT_directory_entryFromPath ( state->partition, &dirEntry, path, NULL );
|
fileExists = _FAT_directory_entryFromPath(state->partition, &dirEntry, path, NULL);
|
||||||
|
|
||||||
if ( !fileExists )
|
if (!fileExists)
|
||||||
{
|
{
|
||||||
_FAT_unlock( &state->partition->lock );
|
_FAT_unlock(&state->partition->lock);
|
||||||
r->_errno = ENOENT;
|
r->_errno = ENOENT;
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make sure it is a directory
|
// Make sure it is a directory
|
||||||
if ( ! _FAT_directory_isDirectory ( &dirEntry ) )
|
if (!_FAT_directory_isDirectory(&dirEntry))
|
||||||
{
|
{
|
||||||
_FAT_unlock( &state->partition->lock );
|
_FAT_unlock(&state->partition->lock);
|
||||||
r->_errno = ENOTDIR;
|
r->_errno = ENOTDIR;
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Save the start cluster for use when resetting the directory data
|
// Save the start cluster for use when resetting the directory data
|
||||||
state->startCluster = _FAT_directory_entryGetCluster ( state->partition, dirEntry.entryData );
|
state->startCluster = _FAT_directory_entryGetCluster(state->partition, dirEntry.entryData);
|
||||||
|
|
||||||
// Get the first entry for use with a call to dirnext
|
// Get the first entry for use with a call to dirnext
|
||||||
state->validEntry =
|
state->validEntry = _FAT_directory_getFirstEntry(state->partition, &(state->currentEntry), state->startCluster);
|
||||||
_FAT_directory_getFirstEntry ( state->partition, &( state->currentEntry ), state->startCluster );
|
|
||||||
|
|
||||||
// We are now using this entry
|
// We are now using this entry
|
||||||
state->inUse = true;
|
state->inUse = true;
|
||||||
_FAT_unlock( &state->partition->lock );
|
_FAT_unlock(&state->partition->lock);
|
||||||
return ( DIR_ITER* ) state;
|
return (DIR_ITER*) state;
|
||||||
}
|
}
|
||||||
|
|
||||||
int _FAT_dirreset_r ( struct _reent *r, DIR_ITER *dirState )
|
int _FAT_dirreset_r(struct _reent *r, DIR_ITER *dirState)
|
||||||
{
|
{
|
||||||
DIR_STATE_STRUCT* state = ( DIR_STATE_STRUCT* ) ( dirState->dirStruct );
|
DIR_STATE_STRUCT* state = (DIR_STATE_STRUCT*) (dirState->dirStruct);
|
||||||
|
|
||||||
_FAT_lock( &state->partition->lock );
|
_FAT_lock(&state->partition->lock);
|
||||||
|
|
||||||
// Make sure we are still using this entry
|
// Make sure we are still using this entry
|
||||||
if ( !state->inUse )
|
if (!state->inUse)
|
||||||
{
|
{
|
||||||
_FAT_unlock( &state->partition->lock );
|
_FAT_unlock(&state->partition->lock);
|
||||||
r->_errno = EBADF;
|
r->_errno = EBADF;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the first entry for use with a call to dirnext
|
// Get the first entry for use with a call to dirnext
|
||||||
state->validEntry =
|
state->validEntry = _FAT_directory_getFirstEntry(state->partition, &(state->currentEntry), state->startCluster);
|
||||||
_FAT_directory_getFirstEntry ( state->partition, &( state->currentEntry ), state->startCluster );
|
|
||||||
|
|
||||||
_FAT_unlock( &state->partition->lock );
|
_FAT_unlock(&state->partition->lock);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int _FAT_dirnext_r ( struct _reent *r, DIR_ITER *dirState, char *filename, struct stat *filestat )
|
int _FAT_dirnext_r(struct _reent *r, DIR_ITER *dirState, char *filename, struct stat *filestat)
|
||||||
{
|
{
|
||||||
DIR_STATE_STRUCT* state = ( DIR_STATE_STRUCT* ) ( dirState->dirStruct );
|
DIR_STATE_STRUCT* state = (DIR_STATE_STRUCT*) (dirState->dirStruct);
|
||||||
|
|
||||||
_FAT_lock( &state->partition->lock );
|
_FAT_lock(&state->partition->lock);
|
||||||
|
|
||||||
// Make sure we are still using this entry
|
// Make sure we are still using this entry
|
||||||
if ( !state->inUse )
|
if (!state->inUse)
|
||||||
{
|
{
|
||||||
_FAT_unlock( &state->partition->lock );
|
_FAT_unlock(&state->partition->lock);
|
||||||
r->_errno = EBADF;
|
r->_errno = EBADF;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make sure there is another file to report on
|
// Make sure there is another file to report on
|
||||||
if ( ! state->validEntry )
|
if (!state->validEntry)
|
||||||
{
|
{
|
||||||
_FAT_unlock( &state->partition->lock );
|
_FAT_unlock(&state->partition->lock);
|
||||||
r->_errno = ENOENT;
|
r->_errno = ENOENT;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the filename
|
// Get the filename
|
||||||
strncpy ( filename, state->currentEntry.filename, MAX_FILENAME_LENGTH );
|
strncpy(filename, state->currentEntry.filename, MAX_FILENAME_LENGTH);
|
||||||
// Get the stats, if requested
|
// Get the stats, if requested
|
||||||
if ( filestat != NULL )
|
if (filestat != NULL)
|
||||||
{
|
{
|
||||||
_FAT_directory_entryStat ( state->partition, &( state->currentEntry ), filestat );
|
_FAT_directory_entryStat(state->partition, &(state->currentEntry), filestat);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Look for the next entry for use next time
|
// Look for the next entry for use next time
|
||||||
state->validEntry =
|
state->validEntry = _FAT_directory_getNextEntry(state->partition, &(state->currentEntry));
|
||||||
_FAT_directory_getNextEntry ( state->partition, &( state->currentEntry ) );
|
|
||||||
|
|
||||||
_FAT_unlock( &state->partition->lock );
|
_FAT_unlock(&state->partition->lock);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int _FAT_dirclose_r ( struct _reent *r, DIR_ITER *dirState )
|
int _FAT_dirclose_r(struct _reent *r, DIR_ITER *dirState)
|
||||||
{
|
{
|
||||||
DIR_STATE_STRUCT* state = ( DIR_STATE_STRUCT* ) ( dirState->dirStruct );
|
DIR_STATE_STRUCT* state = (DIR_STATE_STRUCT*) (dirState->dirStruct);
|
||||||
|
|
||||||
// We are no longer using this entry
|
// We are no longer using this entry
|
||||||
_FAT_lock( &state->partition->lock );
|
_FAT_lock(&state->partition->lock);
|
||||||
state->inUse = false;
|
state->inUse = false;
|
||||||
_FAT_unlock( &state->partition->lock );
|
_FAT_unlock(&state->partition->lock);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -9,13 +9,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -26,8 +26,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
#ifndef __FATDIR_H
|
#ifndef __FATDIR_H
|
||||||
#define __FATDIR_H
|
#define __FATDIR_H
|
||||||
@ -41,34 +40,33 @@
|
|||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
PARTITION* partition;
|
PARTITION* partition;
|
||||||
DIR_ENTRY currentEntry;
|
DIR_ENTRY currentEntry;
|
||||||
uint32_t startCluster;
|
uint32_t startCluster;
|
||||||
bool inUse;
|
bool inUse;
|
||||||
bool validEntry;
|
bool validEntry;
|
||||||
} DIR_STATE_STRUCT;
|
} DIR_STATE_STRUCT;
|
||||||
|
|
||||||
extern int _FAT_stat_r ( struct _reent *r, const char *path, struct stat *st );
|
extern int _FAT_stat_r(struct _reent *r, const char *path, struct stat *st);
|
||||||
|
|
||||||
extern int _FAT_link_r ( struct _reent *r, const char *existing, const char *newLink );
|
extern int _FAT_link_r(struct _reent *r, const char *existing, const char *newLink);
|
||||||
|
|
||||||
extern int _FAT_unlink_r ( struct _reent *r, const char *name );
|
extern int _FAT_unlink_r(struct _reent *r, const char *name);
|
||||||
|
|
||||||
extern int _FAT_chdir_r ( struct _reent *r, const char *name );
|
extern int _FAT_chdir_r(struct _reent *r, const char *name);
|
||||||
|
|
||||||
extern int _FAT_rename_r ( struct _reent *r, const char *oldName, const char *newName );
|
extern int _FAT_rename_r(struct _reent *r, const char *oldName, const char *newName);
|
||||||
|
|
||||||
extern int _FAT_mkdir_r ( struct _reent *r, const char *path, int mode );
|
extern int _FAT_mkdir_r(struct _reent *r, const char *path, int mode);
|
||||||
|
|
||||||
extern int _FAT_statvfs_r ( struct _reent *r, const char *path, struct statvfs *buf );
|
extern int _FAT_statvfs_r(struct _reent *r, const char *path, struct statvfs *buf);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Directory iterator functions
|
Directory iterator functions
|
||||||
*/
|
*/
|
||||||
extern DIR_ITER* _FAT_diropen_r( struct _reent *r, DIR_ITER *dirState, const char *path );
|
extern DIR_ITER* _FAT_diropen_r(struct _reent *r, DIR_ITER *dirState, const char *path);
|
||||||
extern int _FAT_dirreset_r ( struct _reent *r, DIR_ITER *dirState );
|
extern int _FAT_dirreset_r(struct _reent *r, DIR_ITER *dirState);
|
||||||
extern int _FAT_dirnext_r ( struct _reent *r, DIR_ITER *dirState, char *filename, struct stat *filestat );
|
extern int _FAT_dirnext_r(struct _reent *r, DIR_ITER *dirState, char *filename, struct stat *filestat);
|
||||||
extern int _FAT_dirclose_r ( struct _reent *r, DIR_ITER *dirState );
|
extern int _FAT_dirclose_r(struct _reent *r, DIR_ITER *dirState);
|
||||||
|
|
||||||
|
|
||||||
#endif // _FATDIR_H
|
#endif // _FATDIR_H
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -9,13 +9,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -26,8 +26,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
#ifndef _FATFILE_H
|
#ifndef _FATFILE_H
|
||||||
#define _FATFILE_H
|
#define _FATFILE_H
|
||||||
@ -40,68 +39,67 @@
|
|||||||
#include "directory.h"
|
#include "directory.h"
|
||||||
|
|
||||||
#define FILE_MAX_SIZE ((uint32_t)0xFFFFFFFF) // 4GiB - 1B
|
#define FILE_MAX_SIZE ((uint32_t)0xFFFFFFFF) // 4GiB - 1B
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
u32 cluster;
|
u32 cluster;
|
||||||
sec_t sector;
|
sec_t sector;
|
||||||
s32 byte;
|
s32 byte;
|
||||||
} FILE_POSITION;
|
} FILE_POSITION;
|
||||||
|
|
||||||
struct _FILE_STRUCT;
|
struct _FILE_STRUCT;
|
||||||
|
|
||||||
struct _FILE_STRUCT
|
struct _FILE_STRUCT
|
||||||
{
|
{
|
||||||
uint32_t filesize;
|
uint32_t filesize;
|
||||||
uint32_t startCluster;
|
uint32_t startCluster;
|
||||||
uint32_t currentPosition;
|
uint32_t currentPosition;
|
||||||
FILE_POSITION rwPosition;
|
FILE_POSITION rwPosition;
|
||||||
FILE_POSITION appendPosition;
|
FILE_POSITION appendPosition;
|
||||||
DIR_ENTRY_POSITION dirEntryStart; // Points to the start of the LFN entries of a file, or the alias for no LFN
|
DIR_ENTRY_POSITION dirEntryStart; // Points to the start of the LFN entries of a file, or the alias for no LFN
|
||||||
DIR_ENTRY_POSITION dirEntryEnd; // Always points to the file's alias entry
|
DIR_ENTRY_POSITION dirEntryEnd; // Always points to the file's alias entry
|
||||||
PARTITION* partition;
|
PARTITION* partition;
|
||||||
struct _FILE_STRUCT* prevOpenFile; // The previous entry in a double-linked list of open files
|
struct _FILE_STRUCT* prevOpenFile; // The previous entry in a double-linked list of open files
|
||||||
struct _FILE_STRUCT* nextOpenFile; // The next entry in a double-linked list of open files
|
struct _FILE_STRUCT* nextOpenFile; // The next entry in a double-linked list of open files
|
||||||
bool read;
|
bool read;
|
||||||
bool write;
|
bool write;
|
||||||
bool append;
|
bool append;
|
||||||
bool inUse;
|
bool inUse;
|
||||||
bool modified;
|
bool modified;
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef struct _FILE_STRUCT FILE_STRUCT;
|
typedef struct _FILE_STRUCT FILE_STRUCT;
|
||||||
|
|
||||||
extern int _FAT_open_r ( struct _reent *r, void *fileStruct, const char *path, int flags, int mode );
|
extern int _FAT_open_r(struct _reent *r, void *fileStruct, const char *path, int flags, int mode);
|
||||||
|
|
||||||
extern int _FAT_close_r ( struct _reent *r, int fd );
|
extern int _FAT_close_r(struct _reent *r, int fd);
|
||||||
|
|
||||||
extern ssize_t _FAT_write_r ( struct _reent *r, int fd, const char *ptr, size_t len );
|
extern ssize_t _FAT_write_r(struct _reent *r, int fd, const char *ptr, size_t len);
|
||||||
|
|
||||||
extern ssize_t _FAT_read_r ( struct _reent *r, int fd, char *ptr, size_t len );
|
extern ssize_t _FAT_read_r(struct _reent *r, int fd, char *ptr, size_t len);
|
||||||
|
|
||||||
extern off_t _FAT_seek_r ( struct _reent *r, int fd, off_t pos, int dir );
|
extern off_t _FAT_seek_r(struct _reent *r, int fd, off_t pos, int dir);
|
||||||
|
|
||||||
extern int _FAT_fstat_r ( struct _reent *r, int fd, struct stat *st );
|
extern int _FAT_fstat_r(struct _reent *r, int fd, struct stat *st);
|
||||||
|
|
||||||
extern int _FAT_stat_r ( struct _reent *r, const char *path, struct stat *st );
|
extern int _FAT_stat_r(struct _reent *r, const char *path, struct stat *st);
|
||||||
|
|
||||||
extern int _FAT_link_r ( struct _reent *r, const char *existing, const char *newLink );
|
extern int _FAT_link_r(struct _reent *r, const char *existing, const char *newLink);
|
||||||
|
|
||||||
extern int _FAT_unlink_r ( struct _reent *r, const char *name );
|
extern int _FAT_unlink_r(struct _reent *r, const char *name);
|
||||||
|
|
||||||
extern int _FAT_chdir_r ( struct _reent *r, const char *name );
|
extern int _FAT_chdir_r(struct _reent *r, const char *name);
|
||||||
|
|
||||||
extern int _FAT_rename_r ( struct _reent *r, const char *oldName, const char *newName );
|
extern int _FAT_rename_r(struct _reent *r, const char *oldName, const char *newName);
|
||||||
|
|
||||||
extern int _FAT_ftruncate_r ( struct _reent *r, int fd, off_t len );
|
extern int _FAT_ftruncate_r(struct _reent *r, int fd, off_t len);
|
||||||
|
|
||||||
extern int _FAT_fsync_r ( struct _reent *r, int fd );
|
extern int _FAT_fsync_r(struct _reent *r, int fd);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Synchronizes the file data to disc.
|
Synchronizes the file data to disc.
|
||||||
Does no locking of its own -- lock the partition before calling.
|
Does no locking of its own -- lock the partition before calling.
|
||||||
Returns 0 on success, an error code on failure.
|
Returns 0 on success, an error code on failure.
|
||||||
*/
|
*/
|
||||||
extern int _FAT_syncToDisc ( FILE_STRUCT* file );
|
extern int _FAT_syncToDisc(FILE_STRUCT* file);
|
||||||
|
|
||||||
#endif // _FATFILE_H
|
#endif // _FATFILE_H
|
||||||
|
@ -8,13 +8,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -25,89 +25,87 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
#include "file_allocation_table.h"
|
#include "file_allocation_table.h"
|
||||||
#include "partition.h"
|
#include "partition.h"
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Gets the cluster linked from input cluster
|
Gets the cluster linked from input cluster
|
||||||
*/
|
*/
|
||||||
uint32_t _FAT_fat_nextCluster( PARTITION* partition, uint32_t cluster )
|
uint32_t _FAT_fat_nextCluster(PARTITION* partition, uint32_t cluster)
|
||||||
{
|
{
|
||||||
uint32_t nextCluster = CLUSTER_FREE;
|
uint32_t nextCluster = CLUSTER_FREE;
|
||||||
sec_t sector;
|
sec_t sector;
|
||||||
int offset;
|
int offset;
|
||||||
|
|
||||||
if ( cluster == CLUSTER_FREE )
|
if (cluster == CLUSTER_FREE)
|
||||||
{
|
{
|
||||||
return CLUSTER_FREE;
|
return CLUSTER_FREE;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ( partition->filesysType )
|
switch (partition->filesysType)
|
||||||
{
|
{
|
||||||
case FS_UNKNOWN:
|
case FS_UNKNOWN:
|
||||||
return CLUSTER_ERROR;
|
return CLUSTER_ERROR;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case FS_FAT12:
|
case FS_FAT12:
|
||||||
|
{
|
||||||
|
u32 nextCluster_h;
|
||||||
|
sector = partition->fat.fatStart + (((cluster * 3) / 2) / BYTES_PER_READ);
|
||||||
|
offset = ((cluster * 3) / 2) % BYTES_PER_READ;
|
||||||
|
|
||||||
|
_FAT_cache_readLittleEndianValue(partition->cache, &nextCluster, sector, offset, sizeof(u8));
|
||||||
|
|
||||||
|
offset++;
|
||||||
|
|
||||||
|
if (offset >= BYTES_PER_READ)
|
||||||
{
|
{
|
||||||
u32 nextCluster_h;
|
offset = 0;
|
||||||
sector = partition->fat.fatStart + ( ( ( cluster * 3 ) / 2 ) / BYTES_PER_READ );
|
sector++;
|
||||||
offset = ( ( cluster * 3 ) / 2 ) % BYTES_PER_READ;
|
|
||||||
|
|
||||||
|
|
||||||
_FAT_cache_readLittleEndianValue ( partition->cache, &nextCluster, sector, offset, sizeof( u8 ) );
|
|
||||||
|
|
||||||
offset++;
|
|
||||||
|
|
||||||
if ( offset >= BYTES_PER_READ )
|
|
||||||
{
|
|
||||||
offset = 0;
|
|
||||||
sector++;
|
|
||||||
}
|
|
||||||
nextCluster_h = 0;
|
|
||||||
|
|
||||||
_FAT_cache_readLittleEndianValue ( partition->cache, &nextCluster_h, sector, offset, sizeof( u8 ) );
|
|
||||||
nextCluster |= ( nextCluster_h << 8 );
|
|
||||||
|
|
||||||
if ( cluster & 0x01 )
|
|
||||||
{
|
|
||||||
nextCluster = nextCluster >> 4;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
nextCluster &= 0x0FFF;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( nextCluster >= 0x0FF7 )
|
|
||||||
{
|
|
||||||
nextCluster = CLUSTER_EOF;
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
nextCluster_h = 0;
|
||||||
|
|
||||||
|
_FAT_cache_readLittleEndianValue(partition->cache, &nextCluster_h, sector, offset, sizeof(u8));
|
||||||
|
nextCluster |= (nextCluster_h << 8);
|
||||||
|
|
||||||
|
if (cluster & 0x01)
|
||||||
|
{
|
||||||
|
nextCluster = nextCluster >> 4;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
nextCluster &= 0x0FFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (nextCluster >= 0x0FF7)
|
||||||
|
{
|
||||||
|
nextCluster = CLUSTER_EOF;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
case FS_FAT16:
|
case FS_FAT16:
|
||||||
sector = partition->fat.fatStart + ( ( cluster << 1 ) / BYTES_PER_READ );
|
sector = partition->fat.fatStart + ((cluster << 1) / BYTES_PER_READ);
|
||||||
offset = ( cluster % ( BYTES_PER_READ >> 1 ) ) << 1;
|
offset = (cluster % (BYTES_PER_READ >> 1)) << 1;
|
||||||
|
|
||||||
_FAT_cache_readLittleEndianValue ( partition->cache, &nextCluster, sector, offset, sizeof( u16 ) );
|
_FAT_cache_readLittleEndianValue(partition->cache, &nextCluster, sector, offset, sizeof(u16));
|
||||||
|
|
||||||
if ( nextCluster >= 0xFFF7 )
|
if (nextCluster >= 0xFFF7)
|
||||||
{
|
{
|
||||||
nextCluster = CLUSTER_EOF;
|
nextCluster = CLUSTER_EOF;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case FS_FAT32:
|
case FS_FAT32:
|
||||||
sector = partition->fat.fatStart + ( ( cluster << 2 ) / BYTES_PER_READ );
|
sector = partition->fat.fatStart + ((cluster << 2) / BYTES_PER_READ);
|
||||||
offset = ( cluster % ( BYTES_PER_READ >> 2 ) ) << 2;
|
offset = (cluster % (BYTES_PER_READ >> 2)) << 2;
|
||||||
|
|
||||||
_FAT_cache_readLittleEndianValue ( partition->cache, &nextCluster, sector, offset, sizeof( u32 ) );
|
_FAT_cache_readLittleEndianValue(partition->cache, &nextCluster, sector, offset, sizeof(u32));
|
||||||
|
|
||||||
if ( nextCluster >= 0x0FFFFFF7 )
|
if (nextCluster >= 0x0FFFFFF7)
|
||||||
{
|
{
|
||||||
nextCluster = CLUSTER_EOF;
|
nextCluster = CLUSTER_EOF;
|
||||||
}
|
}
|
||||||
@ -122,83 +120,83 @@ uint32_t _FAT_fat_nextCluster( PARTITION* partition, uint32_t cluster )
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
writes value into the correct offset within a partition's FAT, based
|
writes value into the correct offset within a partition's FAT, based
|
||||||
on the cluster number.
|
on the cluster number.
|
||||||
*/
|
*/
|
||||||
static bool _FAT_fat_writeFatEntry ( PARTITION* partition, uint32_t cluster, uint32_t value )
|
static bool _FAT_fat_writeFatEntry(PARTITION* partition, uint32_t cluster, uint32_t value)
|
||||||
{
|
{
|
||||||
sec_t sector;
|
sec_t sector;
|
||||||
int offset;
|
int offset;
|
||||||
uint32_t oldValue;
|
uint32_t oldValue;
|
||||||
|
|
||||||
if ( ( cluster < CLUSTER_FIRST ) || ( cluster > partition->fat.lastCluster /* This will catch CLUSTER_ERROR */ ) )
|
if ((cluster < CLUSTER_FIRST) || (cluster > partition->fat.lastCluster /* This will catch CLUSTER_ERROR */))
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch ( partition->filesysType )
|
switch (partition->filesysType)
|
||||||
{
|
{
|
||||||
case FS_UNKNOWN:
|
case FS_UNKNOWN:
|
||||||
return false;
|
return false;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case FS_FAT12:
|
case FS_FAT12:
|
||||||
sector = partition->fat.fatStart + ( ( ( cluster * 3 ) / 2 ) / BYTES_PER_READ );
|
sector = partition->fat.fatStart + (((cluster * 3) / 2) / BYTES_PER_READ);
|
||||||
offset = ( ( cluster * 3 ) / 2 ) % BYTES_PER_READ;
|
offset = ((cluster * 3) / 2) % BYTES_PER_READ;
|
||||||
|
|
||||||
if ( cluster & 0x01 )
|
if (cluster & 0x01)
|
||||||
{
|
{
|
||||||
|
|
||||||
_FAT_cache_readLittleEndianValue ( partition->cache, &oldValue, sector, offset, sizeof( u8 ) );
|
_FAT_cache_readLittleEndianValue(partition->cache, &oldValue, sector, offset, sizeof(u8));
|
||||||
|
|
||||||
value = ( value << 4 ) | ( oldValue & 0x0F );
|
value = (value << 4) | (oldValue & 0x0F);
|
||||||
|
|
||||||
_FAT_cache_writeLittleEndianValue ( partition->cache, value & 0xFF, sector, offset, sizeof( u8 ) );
|
_FAT_cache_writeLittleEndianValue(partition->cache, value & 0xFF, sector, offset, sizeof(u8));
|
||||||
|
|
||||||
offset++;
|
offset++;
|
||||||
if ( offset >= BYTES_PER_READ )
|
if (offset >= BYTES_PER_READ)
|
||||||
{
|
{
|
||||||
offset = 0;
|
offset = 0;
|
||||||
sector++;
|
sector++;
|
||||||
}
|
}
|
||||||
|
|
||||||
_FAT_cache_writeLittleEndianValue ( partition->cache, ( value >> 8 ) & 0xFF, sector, offset, sizeof( u8 ) );
|
_FAT_cache_writeLittleEndianValue(partition->cache, (value >> 8) & 0xFF, sector, offset, sizeof(u8));
|
||||||
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
|
||||||
_FAT_cache_writeLittleEndianValue ( partition->cache, value, sector, offset, sizeof( u8 ) );
|
_FAT_cache_writeLittleEndianValue(partition->cache, value, sector, offset, sizeof(u8));
|
||||||
|
|
||||||
offset++;
|
offset++;
|
||||||
if ( offset >= BYTES_PER_READ )
|
if (offset >= BYTES_PER_READ)
|
||||||
{
|
{
|
||||||
offset = 0;
|
offset = 0;
|
||||||
sector++;
|
sector++;
|
||||||
}
|
}
|
||||||
|
|
||||||
_FAT_cache_readLittleEndianValue ( partition->cache, &oldValue, sector, offset, sizeof( u8 ) );
|
_FAT_cache_readLittleEndianValue(partition->cache, &oldValue, sector, offset, sizeof(u8));
|
||||||
|
|
||||||
value = ( ( value >> 8 ) & 0x0F ) | ( oldValue & 0xF0 );
|
value = ((value >> 8) & 0x0F) | (oldValue & 0xF0);
|
||||||
|
|
||||||
_FAT_cache_writeLittleEndianValue ( partition->cache, value, sector, offset, sizeof( u8 ) );
|
_FAT_cache_writeLittleEndianValue(partition->cache, value, sector, offset, sizeof(u8));
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case FS_FAT16:
|
case FS_FAT16:
|
||||||
sector = partition->fat.fatStart + ( ( cluster << 1 ) / BYTES_PER_READ );
|
sector = partition->fat.fatStart + ((cluster << 1) / BYTES_PER_READ);
|
||||||
offset = ( cluster % ( BYTES_PER_READ >> 1 ) ) << 1;
|
offset = (cluster % (BYTES_PER_READ >> 1)) << 1;
|
||||||
|
|
||||||
_FAT_cache_writeLittleEndianValue ( partition->cache, value, sector, offset, sizeof( u16 ) );
|
_FAT_cache_writeLittleEndianValue(partition->cache, value, sector, offset, sizeof(u16));
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case FS_FAT32:
|
case FS_FAT32:
|
||||||
sector = partition->fat.fatStart + ( ( cluster << 2 ) / BYTES_PER_READ );
|
sector = partition->fat.fatStart + ((cluster << 2) / BYTES_PER_READ);
|
||||||
offset = ( cluster % ( BYTES_PER_READ >> 2 ) ) << 2;
|
offset = (cluster % (BYTES_PER_READ >> 2)) << 2;
|
||||||
|
|
||||||
_FAT_cache_writeLittleEndianValue ( partition->cache, value, sector, offset, sizeof( u32 ) );
|
_FAT_cache_writeLittleEndianValue(partition->cache, value, sector, offset, sizeof(u32));
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -211,28 +209,28 @@ static bool _FAT_fat_writeFatEntry ( PARTITION* partition, uint32_t cluster, uin
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*-----------------------------------------------------------------
|
/*-----------------------------------------------------------------
|
||||||
gets the first available free cluster, sets it
|
gets the first available free cluster, sets it
|
||||||
to end of file, links the input cluster to it then returns the
|
to end of file, links the input cluster to it then returns the
|
||||||
cluster number
|
cluster number
|
||||||
If an error occurs, return CLUSTER_ERROR
|
If an error occurs, return CLUSTER_ERROR
|
||||||
-----------------------------------------------------------------*/
|
-----------------------------------------------------------------*/
|
||||||
uint32_t _FAT_fat_linkFreeCluster( PARTITION* partition, uint32_t cluster )
|
uint32_t _FAT_fat_linkFreeCluster(PARTITION* partition, uint32_t cluster)
|
||||||
{
|
{
|
||||||
uint32_t firstFree;
|
uint32_t firstFree;
|
||||||
uint32_t curLink;
|
uint32_t curLink;
|
||||||
uint32_t lastCluster;
|
uint32_t lastCluster;
|
||||||
bool loopedAroundFAT = false;
|
bool loopedAroundFAT = false;
|
||||||
|
|
||||||
lastCluster = partition->fat.lastCluster;
|
lastCluster = partition->fat.lastCluster;
|
||||||
|
|
||||||
if ( cluster > lastCluster )
|
if (cluster > lastCluster)
|
||||||
{
|
{
|
||||||
return CLUSTER_ERROR;
|
return CLUSTER_ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if the cluster already has a link, and return it if so
|
// Check if the cluster already has a link, and return it if so
|
||||||
curLink = _FAT_fat_nextCluster( partition, cluster );
|
curLink = _FAT_fat_nextCluster(partition, cluster);
|
||||||
if ( ( curLink >= CLUSTER_FIRST ) && ( curLink <= lastCluster ) )
|
if ((curLink >= CLUSTER_FIRST) && (curLink <= lastCluster))
|
||||||
{
|
{
|
||||||
return curLink; // Return the current link - don't allocate a new one
|
return curLink; // Return the current link - don't allocate a new one
|
||||||
}
|
}
|
||||||
@ -240,18 +238,18 @@ uint32_t _FAT_fat_linkFreeCluster( PARTITION* partition, uint32_t cluster )
|
|||||||
// Get a free cluster
|
// Get a free cluster
|
||||||
firstFree = partition->fat.firstFree;
|
firstFree = partition->fat.firstFree;
|
||||||
// Start at first valid cluster
|
// Start at first valid cluster
|
||||||
if ( firstFree < CLUSTER_FIRST )
|
if (firstFree < CLUSTER_FIRST)
|
||||||
{
|
{
|
||||||
firstFree = CLUSTER_FIRST;
|
firstFree = CLUSTER_FIRST;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Search until a free cluster is found
|
// Search until a free cluster is found
|
||||||
while ( _FAT_fat_nextCluster( partition, firstFree ) != CLUSTER_FREE )
|
while (_FAT_fat_nextCluster(partition, firstFree) != CLUSTER_FREE)
|
||||||
{
|
{
|
||||||
firstFree++;
|
firstFree++;
|
||||||
if ( firstFree > lastCluster )
|
if (firstFree > lastCluster)
|
||||||
{
|
{
|
||||||
if ( loopedAroundFAT )
|
if (loopedAroundFAT)
|
||||||
{
|
{
|
||||||
// If couldn't get a free cluster then return an error
|
// If couldn't get a free cluster then return an error
|
||||||
partition->fat.firstFree = firstFree;
|
partition->fat.firstFree = firstFree;
|
||||||
@ -268,74 +266,70 @@ uint32_t _FAT_fat_linkFreeCluster( PARTITION* partition, uint32_t cluster )
|
|||||||
}
|
}
|
||||||
partition->fat.firstFree = firstFree;
|
partition->fat.firstFree = firstFree;
|
||||||
|
|
||||||
if ( ( cluster >= CLUSTER_FIRST ) && ( cluster < lastCluster ) )
|
if ((cluster >= CLUSTER_FIRST) && (cluster < lastCluster))
|
||||||
{
|
{
|
||||||
// Update the linked from FAT entry
|
// Update the linked from FAT entry
|
||||||
_FAT_fat_writeFatEntry ( partition, cluster, firstFree );
|
_FAT_fat_writeFatEntry(partition, cluster, firstFree);
|
||||||
}
|
}
|
||||||
// Create the linked to FAT entry
|
// Create the linked to FAT entry
|
||||||
_FAT_fat_writeFatEntry ( partition, firstFree, CLUSTER_EOF );
|
_FAT_fat_writeFatEntry(partition, firstFree, CLUSTER_EOF);
|
||||||
|
|
||||||
return firstFree;
|
return firstFree;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*-----------------------------------------------------------------
|
/*-----------------------------------------------------------------
|
||||||
gets the first available free cluster, sets it
|
gets the first available free cluster, sets it
|
||||||
to end of file, links the input cluster to it, clears the new
|
to end of file, links the input cluster to it, clears the new
|
||||||
cluster to 0 valued bytes, then returns the cluster number
|
cluster to 0 valued bytes, then returns the cluster number
|
||||||
If an error occurs, return CLUSTER_ERROR
|
If an error occurs, return CLUSTER_ERROR
|
||||||
-----------------------------------------------------------------*/
|
-----------------------------------------------------------------*/
|
||||||
uint32_t _FAT_fat_linkFreeClusterCleared ( PARTITION* partition, uint32_t cluster )
|
uint32_t _FAT_fat_linkFreeClusterCleared(PARTITION* partition, uint32_t cluster)
|
||||||
{
|
{
|
||||||
uint32_t newCluster;
|
uint32_t newCluster;
|
||||||
uint32_t i;
|
uint32_t i;
|
||||||
uint8_t emptySector[BYTES_PER_READ];
|
uint8_t emptySector[BYTES_PER_READ];
|
||||||
|
|
||||||
// Link the cluster
|
// Link the cluster
|
||||||
newCluster = _FAT_fat_linkFreeCluster( partition, cluster );
|
newCluster = _FAT_fat_linkFreeCluster(partition, cluster);
|
||||||
|
|
||||||
if ( newCluster == CLUSTER_FREE || newCluster == CLUSTER_ERROR )
|
if (newCluster == CLUSTER_FREE || newCluster == CLUSTER_ERROR)
|
||||||
{
|
{
|
||||||
return CLUSTER_ERROR;
|
return CLUSTER_ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Clear all the sectors within the cluster
|
// Clear all the sectors within the cluster
|
||||||
memset ( emptySector, 0, BYTES_PER_READ );
|
memset(emptySector, 0, BYTES_PER_READ);
|
||||||
for ( i = 0; i < partition->sectorsPerCluster; i++ )
|
for (i = 0; i < partition->sectorsPerCluster; i++)
|
||||||
{
|
{
|
||||||
_FAT_cache_writeSectors ( partition->cache,
|
_FAT_cache_writeSectors(partition->cache, _FAT_fat_clusterToSector(partition, newCluster) + i, 1, emptySector);
|
||||||
_FAT_fat_clusterToSector ( partition, newCluster ) + i,
|
|
||||||
1, emptySector );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return newCluster;
|
return newCluster;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*-----------------------------------------------------------------
|
/*-----------------------------------------------------------------
|
||||||
_FAT_fat_clearLinks
|
_FAT_fat_clearLinks
|
||||||
frees any cluster used by a file
|
frees any cluster used by a file
|
||||||
-----------------------------------------------------------------*/
|
-----------------------------------------------------------------*/
|
||||||
bool _FAT_fat_clearLinks ( PARTITION* partition, uint32_t cluster )
|
bool _FAT_fat_clearLinks(PARTITION* partition, uint32_t cluster)
|
||||||
{
|
{
|
||||||
uint32_t nextCluster;
|
uint32_t nextCluster;
|
||||||
|
|
||||||
if ( ( cluster < CLUSTER_FIRST ) || ( cluster > partition->fat.lastCluster /* This will catch CLUSTER_ERROR */ ) )
|
if ((cluster < CLUSTER_FIRST) || (cluster > partition->fat.lastCluster /* This will catch CLUSTER_ERROR */)) return false;
|
||||||
return false;
|
|
||||||
|
|
||||||
// If this clears up more space in the FAT before the current free pointer, move it backwards
|
// If this clears up more space in the FAT before the current free pointer, move it backwards
|
||||||
if ( cluster < partition->fat.firstFree )
|
if (cluster < partition->fat.firstFree)
|
||||||
{
|
{
|
||||||
partition->fat.firstFree = cluster;
|
partition->fat.firstFree = cluster;
|
||||||
}
|
}
|
||||||
|
|
||||||
while ( ( cluster != CLUSTER_EOF ) && ( cluster != CLUSTER_FREE ) && ( cluster != CLUSTER_ERROR ) )
|
while ((cluster != CLUSTER_EOF) && (cluster != CLUSTER_FREE) && (cluster != CLUSTER_ERROR))
|
||||||
{
|
{
|
||||||
// Store next cluster before erasing the link
|
// Store next cluster before erasing the link
|
||||||
nextCluster = _FAT_fat_nextCluster ( partition, cluster );
|
nextCluster = _FAT_fat_nextCluster(partition, cluster);
|
||||||
|
|
||||||
// Erase the link
|
// Erase the link
|
||||||
_FAT_fat_writeFatEntry ( partition, cluster, CLUSTER_FREE );
|
_FAT_fat_writeFatEntry(partition, cluster, CLUSTER_FREE);
|
||||||
|
|
||||||
// Move onto next cluster
|
// Move onto next cluster
|
||||||
cluster = nextCluster;
|
cluster = nextCluster;
|
||||||
@ -345,73 +339,74 @@ bool _FAT_fat_clearLinks ( PARTITION* partition, uint32_t cluster )
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*-----------------------------------------------------------------
|
/*-----------------------------------------------------------------
|
||||||
_FAT_fat_trimChain
|
_FAT_fat_trimChain
|
||||||
Drop all clusters past the chainLength.
|
Drop all clusters past the chainLength.
|
||||||
If chainLength is 0, all clusters are dropped.
|
If chainLength is 0, all clusters are dropped.
|
||||||
If chainLength is 1, the first cluster is kept and the rest are
|
If chainLength is 1, the first cluster is kept and the rest are
|
||||||
dropped, and so on.
|
dropped, and so on.
|
||||||
Return the last cluster left in the chain.
|
Return the last cluster left in the chain.
|
||||||
-----------------------------------------------------------------*/
|
-----------------------------------------------------------------*/
|
||||||
uint32_t _FAT_fat_trimChain ( PARTITION* partition, uint32_t startCluster, unsigned int chainLength )
|
uint32_t _FAT_fat_trimChain(PARTITION* partition, uint32_t startCluster, unsigned int chainLength)
|
||||||
{
|
{
|
||||||
uint32_t nextCluster;
|
uint32_t nextCluster;
|
||||||
|
|
||||||
if ( chainLength == 0 )
|
if (chainLength == 0)
|
||||||
{
|
{
|
||||||
// Drop the entire chain
|
// Drop the entire chain
|
||||||
_FAT_fat_clearLinks ( partition, startCluster );
|
_FAT_fat_clearLinks(partition, startCluster);
|
||||||
return CLUSTER_FREE;
|
return CLUSTER_FREE;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Find the last cluster in the chain, and the one after it
|
// Find the last cluster in the chain, and the one after it
|
||||||
chainLength--;
|
chainLength--;
|
||||||
nextCluster = _FAT_fat_nextCluster ( partition, startCluster );
|
nextCluster = _FAT_fat_nextCluster(partition, startCluster);
|
||||||
while ( ( chainLength > 0 ) && ( nextCluster != CLUSTER_FREE ) && ( nextCluster != CLUSTER_EOF ) )
|
while ((chainLength > 0) && (nextCluster != CLUSTER_FREE) && (nextCluster != CLUSTER_EOF))
|
||||||
{
|
{
|
||||||
chainLength--;
|
chainLength--;
|
||||||
startCluster = nextCluster;
|
startCluster = nextCluster;
|
||||||
nextCluster = _FAT_fat_nextCluster ( partition, startCluster );
|
nextCluster = _FAT_fat_nextCluster(partition, startCluster);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Drop all clusters after the last in the chain
|
// Drop all clusters after the last in the chain
|
||||||
if ( nextCluster != CLUSTER_FREE && nextCluster != CLUSTER_EOF )
|
if (nextCluster != CLUSTER_FREE && nextCluster != CLUSTER_EOF)
|
||||||
{
|
{
|
||||||
_FAT_fat_clearLinks ( partition, nextCluster );
|
_FAT_fat_clearLinks(partition, nextCluster);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mark the last cluster in the chain as the end of the file
|
// Mark the last cluster in the chain as the end of the file
|
||||||
_FAT_fat_writeFatEntry ( partition, startCluster, CLUSTER_EOF );
|
_FAT_fat_writeFatEntry(partition, startCluster, CLUSTER_EOF);
|
||||||
|
|
||||||
return startCluster;
|
return startCluster;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*-----------------------------------------------------------------
|
/*-----------------------------------------------------------------
|
||||||
_FAT_fat_lastCluster
|
_FAT_fat_lastCluster
|
||||||
Trace the cluster links until the last one is found
|
Trace the cluster links until the last one is found
|
||||||
-----------------------------------------------------------------*/
|
-----------------------------------------------------------------*/
|
||||||
uint32_t _FAT_fat_lastCluster ( PARTITION* partition, uint32_t cluster )
|
uint32_t _FAT_fat_lastCluster(PARTITION* partition, uint32_t cluster)
|
||||||
{
|
{
|
||||||
while ( ( _FAT_fat_nextCluster( partition, cluster ) != CLUSTER_FREE ) && ( _FAT_fat_nextCluster( partition, cluster ) != CLUSTER_EOF ) )
|
while ((_FAT_fat_nextCluster(partition, cluster) != CLUSTER_FREE) && (_FAT_fat_nextCluster(partition, cluster)
|
||||||
|
!= CLUSTER_EOF))
|
||||||
{
|
{
|
||||||
cluster = _FAT_fat_nextCluster( partition, cluster );
|
cluster = _FAT_fat_nextCluster(partition, cluster);
|
||||||
}
|
}
|
||||||
return cluster;
|
return cluster;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*-----------------------------------------------------------------
|
/*-----------------------------------------------------------------
|
||||||
_FAT_fat_freeClusterCount
|
_FAT_fat_freeClusterCount
|
||||||
Return the number of free clusters available
|
Return the number of free clusters available
|
||||||
-----------------------------------------------------------------*/
|
-----------------------------------------------------------------*/
|
||||||
unsigned int _FAT_fat_freeClusterCount ( PARTITION* partition )
|
unsigned int _FAT_fat_freeClusterCount(PARTITION* partition)
|
||||||
{
|
{
|
||||||
unsigned int count = 0;
|
unsigned int count = 0;
|
||||||
uint32_t curCluster;
|
uint32_t curCluster;
|
||||||
|
|
||||||
for ( curCluster = CLUSTER_FIRST; curCluster <= partition->fat.lastCluster; curCluster++ )
|
for (curCluster = CLUSTER_FIRST; curCluster <= partition->fat.lastCluster; curCluster++)
|
||||||
{
|
{
|
||||||
if ( _FAT_fat_nextCluster( partition, curCluster ) == CLUSTER_FREE )
|
if (_FAT_fat_nextCluster(partition, curCluster) == CLUSTER_FREE)
|
||||||
{
|
{
|
||||||
count++;
|
count++;
|
||||||
}
|
}
|
||||||
|
@ -8,13 +8,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -25,7 +25,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __FAT_H
|
#ifndef __FAT_H
|
||||||
#define __FAT_H
|
#define __FAT_H
|
||||||
@ -43,30 +43,28 @@
|
|||||||
#define CLUSTERS_PER_FAT12 4085
|
#define CLUSTERS_PER_FAT12 4085
|
||||||
#define CLUSTERS_PER_FAT16 65525
|
#define CLUSTERS_PER_FAT16 65525
|
||||||
|
|
||||||
|
uint32_t _FAT_fat_nextCluster(PARTITION* partition, uint32_t cluster);
|
||||||
|
|
||||||
uint32_t _FAT_fat_nextCluster( PARTITION* partition, uint32_t cluster );
|
uint32_t _FAT_fat_linkFreeCluster(PARTITION* partition, uint32_t cluster);
|
||||||
|
uint32_t _FAT_fat_linkFreeClusterCleared(PARTITION* partition, uint32_t cluster);
|
||||||
|
|
||||||
uint32_t _FAT_fat_linkFreeCluster( PARTITION* partition, uint32_t cluster );
|
bool _FAT_fat_clearLinks(PARTITION* partition, uint32_t cluster);
|
||||||
uint32_t _FAT_fat_linkFreeClusterCleared ( PARTITION* partition, uint32_t cluster );
|
|
||||||
|
|
||||||
bool _FAT_fat_clearLinks ( PARTITION* partition, uint32_t cluster );
|
uint32_t _FAT_fat_trimChain(PARTITION* partition, uint32_t startCluster, unsigned int chainLength);
|
||||||
|
|
||||||
uint32_t _FAT_fat_trimChain ( PARTITION* partition, uint32_t startCluster, unsigned int chainLength );
|
uint32_t _FAT_fat_lastCluster(PARTITION* partition, uint32_t cluster);
|
||||||
|
|
||||||
uint32_t _FAT_fat_lastCluster ( PARTITION* partition, uint32_t cluster );
|
unsigned int _FAT_fat_freeClusterCount(PARTITION* partition);
|
||||||
|
|
||||||
unsigned int _FAT_fat_freeClusterCount ( PARTITION* partition );
|
static inline sec_t _FAT_fat_clusterToSector(PARTITION* partition, uint32_t cluster)
|
||||||
|
|
||||||
static inline sec_t _FAT_fat_clusterToSector ( PARTITION* partition, uint32_t cluster )
|
|
||||||
{
|
{
|
||||||
return ( cluster >= CLUSTER_FIRST ) ?
|
return (cluster >= CLUSTER_FIRST) ? ((cluster - CLUSTER_FIRST) * (sec_t) partition->sectorsPerCluster)
|
||||||
( ( cluster - CLUSTER_FIRST ) * ( sec_t )partition->sectorsPerCluster ) + partition->dataStart :
|
+ partition->dataStart : partition->rootDirStart;
|
||||||
partition->rootDirStart;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline bool _FAT_fat_isValidCluster ( PARTITION* partition, uint32_t cluster )
|
static inline bool _FAT_fat_isValidCluster(PARTITION* partition, uint32_t cluster)
|
||||||
{
|
{
|
||||||
return ( cluster >= CLUSTER_FIRST ) && ( cluster <= partition->fat.lastCluster /* This will catch CLUSTER_ERROR */ );
|
return (cluster >= CLUSTER_FIRST) && (cluster <= partition->fat.lastCluster /* This will catch CLUSTER_ERROR */);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif // _FAT_H
|
#endif // _FAT_H
|
||||||
|
@ -7,13 +7,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -24,8 +24,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
#include "filetime.h"
|
#include "filetime.h"
|
||||||
@ -40,73 +39,65 @@
|
|||||||
#define MAX_DAY 31
|
#define MAX_DAY 31
|
||||||
#define MIN_DAY 1
|
#define MIN_DAY 1
|
||||||
|
|
||||||
uint16_t _FAT_filetime_getTimeFromRTC ( void )
|
uint16_t _FAT_filetime_getTimeFromRTC(void)
|
||||||
{
|
{
|
||||||
#ifdef USE_RTC_TIME
|
#ifdef USE_RTC_TIME
|
||||||
struct tm timeParts;
|
struct tm timeParts;
|
||||||
time_t epochTime;
|
time_t epochTime;
|
||||||
|
|
||||||
if ( time( &epochTime ) == ( time_t ) - 1 )
|
if (time(&epochTime) == (time_t) -1)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
localtime_r( &epochTime, &timeParts );
|
localtime_r(&epochTime, &timeParts);
|
||||||
|
|
||||||
// Check that the values are all in range.
|
// Check that the values are all in range.
|
||||||
// If they are not, return 0 (no timestamp)
|
// If they are not, return 0 (no timestamp)
|
||||||
if ( ( timeParts.tm_hour < 0 ) || ( timeParts.tm_hour > MAX_HOUR ) ) return 0;
|
if ((timeParts.tm_hour < 0) || (timeParts.tm_hour > MAX_HOUR)) return 0;
|
||||||
if ( ( timeParts.tm_min < 0 ) || ( timeParts.tm_min > MAX_MINUTE ) ) return 0;
|
if ((timeParts.tm_min < 0) || (timeParts.tm_min > MAX_MINUTE)) return 0;
|
||||||
if ( ( timeParts.tm_sec < 0 ) || ( timeParts.tm_sec > MAX_SECOND ) ) return 0;
|
if ((timeParts.tm_sec < 0) || (timeParts.tm_sec > MAX_SECOND)) return 0;
|
||||||
|
|
||||||
return (
|
return (((timeParts.tm_hour & 0x1F) << 11) | ((timeParts.tm_min & 0x3F) << 5) | ((timeParts.tm_sec >> 1) & 0x1F));
|
||||||
( ( timeParts.tm_hour & 0x1F ) << 11 ) |
|
|
||||||
( ( timeParts.tm_min & 0x3F ) << 5 ) |
|
|
||||||
( ( timeParts.tm_sec >> 1 ) & 0x1F )
|
|
||||||
);
|
|
||||||
#else
|
#else
|
||||||
return 0;
|
return 0;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uint16_t _FAT_filetime_getDateFromRTC(void)
|
||||||
uint16_t _FAT_filetime_getDateFromRTC ( void )
|
|
||||||
{
|
{
|
||||||
#ifdef USE_RTC_TIME
|
#ifdef USE_RTC_TIME
|
||||||
struct tm timeParts;
|
struct tm timeParts;
|
||||||
time_t epochTime;
|
time_t epochTime;
|
||||||
|
|
||||||
if ( time( &epochTime ) == ( time_t ) - 1 )
|
if (time(&epochTime) == (time_t) -1)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
localtime_r( &epochTime, &timeParts );
|
localtime_r(&epochTime, &timeParts);
|
||||||
|
|
||||||
if ( ( timeParts.tm_mon < MIN_MONTH ) || ( timeParts.tm_mon > MAX_MONTH ) ) return 0;
|
if ((timeParts.tm_mon < MIN_MONTH) || (timeParts.tm_mon > MAX_MONTH)) return 0;
|
||||||
if ( ( timeParts.tm_mday < MIN_DAY ) || ( timeParts.tm_mday > MAX_DAY ) ) return 0;
|
if ((timeParts.tm_mday < MIN_DAY) || (timeParts.tm_mday > MAX_DAY)) return 0;
|
||||||
|
|
||||||
return (
|
return ((((timeParts.tm_year - 80) & 0x7F) << 9) | // Adjust for MS-FAT base year (1980 vs 1900 for tm_year)
|
||||||
( ( ( timeParts.tm_year - 80 ) & 0x7F ) << 9 ) | // Adjust for MS-FAT base year (1980 vs 1900 for tm_year)
|
(((timeParts.tm_mon + 1) & 0xF) << 5) | (timeParts.tm_mday & 0x1F));
|
||||||
( ( ( timeParts.tm_mon + 1 ) & 0xF ) << 5 ) |
|
|
||||||
( timeParts.tm_mday & 0x1F )
|
|
||||||
);
|
|
||||||
#else
|
#else
|
||||||
return 0;
|
return 0;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
time_t _FAT_filetime_to_time_t ( uint16_t t, uint16_t d )
|
time_t _FAT_filetime_to_time_t(uint16_t t, uint16_t d)
|
||||||
{
|
{
|
||||||
struct tm timeParts;
|
struct tm timeParts;
|
||||||
|
|
||||||
timeParts.tm_hour = t >> 11;
|
timeParts.tm_hour = t >> 11;
|
||||||
timeParts.tm_min = ( t >> 5 ) & 0x3F;
|
timeParts.tm_min = (t >> 5) & 0x3F;
|
||||||
timeParts.tm_sec = ( t & 0x1F ) << 1;
|
timeParts.tm_sec = (t & 0x1F) << 1;
|
||||||
|
|
||||||
timeParts.tm_mday = d & 0x1F;
|
timeParts.tm_mday = d & 0x1F;
|
||||||
timeParts.tm_mon = ( ( d >> 5 ) & 0x0F ) - 1;
|
timeParts.tm_mon = ((d >> 5) & 0x0F) - 1;
|
||||||
timeParts.tm_year = ( d >> 9 ) + 80;
|
timeParts.tm_year = (d >> 9) + 80;
|
||||||
|
|
||||||
timeParts.tm_isdst = 0;
|
timeParts.tm_isdst = 0;
|
||||||
|
|
||||||
return mktime( &timeParts );
|
return mktime(&timeParts);
|
||||||
}
|
}
|
||||||
|
@ -7,13 +7,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -24,7 +24,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __FILETIME_H
|
#ifndef __FILETIME_H
|
||||||
#define __FILETIME_H
|
#define __FILETIME_H
|
||||||
@ -32,10 +32,9 @@
|
|||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
|
|
||||||
uint16_t _FAT_filetime_getTimeFromRTC ( void );
|
uint16_t _FAT_filetime_getTimeFromRTC(void);
|
||||||
uint16_t _FAT_filetime_getDateFromRTC ( void );
|
uint16_t _FAT_filetime_getDateFromRTC(void);
|
||||||
|
|
||||||
time_t _FAT_filetime_to_time_t ( uint16_t t, uint16_t d );
|
|
||||||
|
|
||||||
|
time_t _FAT_filetime_to_time_t(uint16_t t, uint16_t d);
|
||||||
|
|
||||||
#endif // _FILETIME_H
|
#endif // _FILETIME_H
|
||||||
|
@ -1,19 +1,19 @@
|
|||||||
/*
|
/*
|
||||||
libfat.c
|
libfat.c
|
||||||
Simple functionality for startup, mounting and unmounting of FAT-based devices.
|
Simple functionality for startup, mounting and unmounting of FAT-based devices.
|
||||||
|
|
||||||
Copyright (c) 2006 Michael "Chishm" Chisholm
|
Copyright (c) 2006 Michael "Chishm" Chisholm
|
||||||
|
|
||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -24,7 +24,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <sys/iosupport.h>
|
#include <sys/iosupport.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
@ -38,178 +38,153 @@
|
|||||||
#include "mem_allocate.h"
|
#include "mem_allocate.h"
|
||||||
#include "disc_fat.h"
|
#include "disc_fat.h"
|
||||||
|
|
||||||
static const devoptab_t dotab_fat =
|
static const devoptab_t dotab_fat = { "fat", sizeof(FILE_STRUCT), _FAT_open_r, _FAT_close_r, _FAT_write_r, _FAT_read_r,
|
||||||
{
|
_FAT_seek_r, _FAT_fstat_r, _FAT_stat_r, _FAT_link_r, _FAT_unlink_r, _FAT_chdir_r, _FAT_rename_r, _FAT_mkdir_r,
|
||||||
"fat",
|
sizeof(DIR_STATE_STRUCT), _FAT_diropen_r, _FAT_dirreset_r, _FAT_dirnext_r, _FAT_dirclose_r, _FAT_statvfs_r,
|
||||||
sizeof ( FILE_STRUCT ),
|
_FAT_ftruncate_r, _FAT_fsync_r, NULL /* Device data */
|
||||||
_FAT_open_r,
|
|
||||||
_FAT_close_r,
|
|
||||||
_FAT_write_r,
|
|
||||||
_FAT_read_r,
|
|
||||||
_FAT_seek_r,
|
|
||||||
_FAT_fstat_r,
|
|
||||||
_FAT_stat_r,
|
|
||||||
_FAT_link_r,
|
|
||||||
_FAT_unlink_r,
|
|
||||||
_FAT_chdir_r,
|
|
||||||
_FAT_rename_r,
|
|
||||||
_FAT_mkdir_r,
|
|
||||||
sizeof ( DIR_STATE_STRUCT ),
|
|
||||||
_FAT_diropen_r,
|
|
||||||
_FAT_dirreset_r,
|
|
||||||
_FAT_dirnext_r,
|
|
||||||
_FAT_dirclose_r,
|
|
||||||
_FAT_statvfs_r,
|
|
||||||
_FAT_ftruncate_r,
|
|
||||||
_FAT_fsync_r,
|
|
||||||
NULL /* Device data */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
bool fatMount ( const char* name, const DISC_INTERFACE* interface, sec_t startSector, uint32_t cacheSize, uint32_t SectorsPerPage )
|
bool fatMount(const char* name, const DISC_INTERFACE* interface, sec_t startSector, uint32_t cacheSize,
|
||||||
|
uint32_t SectorsPerPage)
|
||||||
{
|
{
|
||||||
PARTITION* partition;
|
PARTITION* partition;
|
||||||
devoptab_t* devops;
|
devoptab_t* devops;
|
||||||
char* nameCopy;
|
char* nameCopy;
|
||||||
|
|
||||||
if ( !interface->startup() )
|
if (!interface->startup()) return false;
|
||||||
return false;
|
|
||||||
|
|
||||||
if ( !interface->isInserted() )
|
if (!interface->isInserted())
|
||||||
{
|
{
|
||||||
interface->shutdown();
|
interface->shutdown();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
devops = _FAT_mem_allocate ( sizeof( devoptab_t ) + strlen( name ) + 1 );
|
devops = _FAT_mem_allocate(sizeof(devoptab_t) + strlen(name) + 1);
|
||||||
if ( !devops )
|
if (!devops)
|
||||||
{
|
{
|
||||||
interface->shutdown();
|
interface->shutdown();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
// Use the space allocated at the end of the devoptab struct for storing the name
|
// Use the space allocated at the end of the devoptab struct for storing the name
|
||||||
nameCopy = ( char* )( devops + 1 );
|
nameCopy = (char*) (devops + 1);
|
||||||
|
|
||||||
// Initialize the file system
|
// Initialize the file system
|
||||||
partition = _FAT_partition_constructor ( interface, cacheSize, SectorsPerPage, startSector );
|
partition = _FAT_partition_constructor(interface, cacheSize, SectorsPerPage, startSector);
|
||||||
if ( !partition )
|
if (!partition)
|
||||||
{
|
{
|
||||||
_FAT_mem_free ( devops );
|
_FAT_mem_free(devops);
|
||||||
interface->shutdown();
|
interface->shutdown();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add an entry for this device to the devoptab table
|
// Add an entry for this device to the devoptab table
|
||||||
memcpy ( devops, &dotab_fat, sizeof( dotab_fat ) );
|
memcpy(devops, &dotab_fat, sizeof(dotab_fat));
|
||||||
strcpy ( nameCopy, name );
|
strcpy(nameCopy, name);
|
||||||
devops->name = nameCopy;
|
devops->name = nameCopy;
|
||||||
devops->deviceData = partition;
|
devops->deviceData = partition;
|
||||||
|
|
||||||
AddDevice ( devops );
|
AddDevice(devops);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool fatMountSimple ( const char* name, const DISC_INTERFACE* interface )
|
bool fatMountSimple(const char* name, const DISC_INTERFACE* interface)
|
||||||
{
|
{
|
||||||
return fatMount ( name, interface, 0, DEFAULT_CACHE_PAGES, DEFAULT_SECTORS_PAGE );
|
return fatMount(name, interface, 0, DEFAULT_CACHE_PAGES, DEFAULT_SECTORS_PAGE);
|
||||||
}
|
}
|
||||||
|
|
||||||
void fatUnmount ( const char* name )
|
void fatUnmount(const char* name)
|
||||||
{
|
{
|
||||||
devoptab_t *devops;
|
devoptab_t *devops;
|
||||||
PARTITION* partition;
|
PARTITION* partition;
|
||||||
const DISC_INTERFACE *disc;
|
const DISC_INTERFACE *disc;
|
||||||
|
|
||||||
devops = ( devoptab_t* )GetDeviceOpTab ( name );
|
devops = (devoptab_t*) GetDeviceOpTab(name);
|
||||||
if ( !devops )
|
if (!devops)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Perform a quick check to make sure we're dealing with a libfat controlled device
|
// Perform a quick check to make sure we're dealing with a libfat controlled device
|
||||||
if ( devops->open_r != dotab_fat.open_r )
|
if (devops->open_r != dotab_fat.open_r)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( RemoveDevice ( name ) == -1 )
|
if (RemoveDevice(name) == -1)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
partition = ( PARTITION* )devops->deviceData;
|
partition = (PARTITION*) devops->deviceData;
|
||||||
disc = partition->disc;
|
disc = partition->disc;
|
||||||
_FAT_partition_destructor ( partition );
|
_FAT_partition_destructor(partition);
|
||||||
_FAT_mem_free ( devops );
|
_FAT_mem_free(devops);
|
||||||
disc->shutdown();
|
disc->shutdown();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool fatInit ( uint32_t cacheSize, bool setAsDefaultDevice )
|
bool fatInit(uint32_t cacheSize, bool setAsDefaultDevice)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
int defaultDevice = -1;
|
int defaultDevice = -1;
|
||||||
const DISC_INTERFACE *disc;
|
const DISC_INTERFACE *disc;
|
||||||
|
|
||||||
for ( i = 0;
|
for (i = 0; _FAT_disc_interfaces[i].name != NULL && _FAT_disc_interfaces[i].getInterface != NULL; i++)
|
||||||
_FAT_disc_interfaces[i].name != NULL && _FAT_disc_interfaces[i].getInterface != NULL;
|
|
||||||
i++ )
|
|
||||||
{
|
{
|
||||||
disc = _FAT_disc_interfaces[i].getInterface();
|
disc = _FAT_disc_interfaces[i].getInterface();
|
||||||
if ( fatMount ( _FAT_disc_interfaces[i].name, disc, 0, cacheSize, DEFAULT_SECTORS_PAGE ) )
|
if (fatMount(_FAT_disc_interfaces[i].name, disc, 0, cacheSize, DEFAULT_SECTORS_PAGE))
|
||||||
{
|
{
|
||||||
// The first device to successfully mount is set as the default
|
// The first device to successfully mount is set as the default
|
||||||
if ( defaultDevice < 0 )
|
if (defaultDevice < 0)
|
||||||
{
|
{
|
||||||
defaultDevice = i;
|
defaultDevice = i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( defaultDevice < 0 )
|
if (defaultDevice < 0)
|
||||||
{
|
{
|
||||||
// None of our devices mounted
|
// None of our devices mounted
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( setAsDefaultDevice )
|
if (setAsDefaultDevice)
|
||||||
{
|
{
|
||||||
char filePath[MAXPATHLEN * 2];
|
char filePath[MAXPATHLEN * 2];
|
||||||
strcpy ( filePath, _FAT_disc_interfaces[defaultDevice].name );
|
strcpy(filePath, _FAT_disc_interfaces[defaultDevice].name);
|
||||||
strcat ( filePath, ":/" );
|
strcat(filePath, ":/");
|
||||||
#ifdef ARGV_MAGIC
|
#ifdef ARGV_MAGIC
|
||||||
if ( __system_argv->argvMagic == ARGV_MAGIC && __system_argv->argc >= 1 && strrchr( __system_argv->argv[0], '/' ) != NULL )
|
if (__system_argv->argvMagic == ARGV_MAGIC && __system_argv->argc >= 1 && strrchr(__system_argv->argv[0], '/')
|
||||||
|
!= NULL)
|
||||||
{
|
{
|
||||||
// Check the app's path against each of our mounted devices, to see
|
// Check the app's path against each of our mounted devices, to see
|
||||||
// if we can support it. If so, change to that path.
|
// if we can support it. If so, change to that path.
|
||||||
for ( i = 0;
|
for (i = 0; _FAT_disc_interfaces[i].name != NULL && _FAT_disc_interfaces[i].getInterface != NULL; i++)
|
||||||
_FAT_disc_interfaces[i].name != NULL && _FAT_disc_interfaces[i].getInterface != NULL;
|
|
||||||
i++ )
|
|
||||||
{
|
{
|
||||||
if ( !strncasecmp( __system_argv->argv[0], _FAT_disc_interfaces[i].name,
|
if (!strncasecmp(__system_argv->argv[0], _FAT_disc_interfaces[i].name, strlen(
|
||||||
strlen( _FAT_disc_interfaces[i].name ) ) )
|
_FAT_disc_interfaces[i].name)))
|
||||||
{
|
{
|
||||||
char *lastSlash;
|
char *lastSlash;
|
||||||
strcpy( filePath, __system_argv->argv[0] );
|
strcpy(filePath, __system_argv->argv[0]);
|
||||||
lastSlash = strrchr( filePath, '/' );
|
lastSlash = strrchr(filePath, '/');
|
||||||
|
|
||||||
if ( NULL != lastSlash )
|
if (NULL != lastSlash)
|
||||||
{
|
{
|
||||||
if ( *( lastSlash - 1 ) == ':' ) lastSlash++;
|
if (*(lastSlash - 1) == ':') lastSlash++;
|
||||||
*lastSlash = 0;
|
*lastSlash = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
chdir ( filePath );
|
chdir(filePath);
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool fatInitDefault ( void )
|
bool fatInitDefault(void)
|
||||||
{
|
{
|
||||||
return fatInit ( DEFAULT_CACHE_PAGES, true );
|
return fatInit(DEFAULT_CACHE_PAGES, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -6,13 +6,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -24,7 +24,7 @@
|
|||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __LOCK_H
|
#ifndef __LOCK_H
|
||||||
#define __LOCK_H
|
#define __LOCK_H
|
||||||
@ -60,28 +60,26 @@ static inline void _FAT_unlock( mutex_t *mutex )
|
|||||||
typedef int mutex_t;
|
typedef int mutex_t;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static inline void _FAT_lock_init( mutex_t *mutex )
|
static inline void _FAT_lock_init(mutex_t *mutex)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void _FAT_lock_deinit( mutex_t *mutex )
|
static inline void _FAT_lock_deinit(mutex_t *mutex)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void _FAT_lock( mutex_t *mutex )
|
static inline void _FAT_lock(mutex_t *mutex)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void _FAT_unlock( mutex_t *mutex )
|
static inline void _FAT_unlock(mutex_t *mutex)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif // USE_LWP_LOCK
|
#endif // USE_LWP_LOCK
|
||||||
|
|
||||||
|
|
||||||
#endif // _LOCK_H
|
#endif // _LOCK_H
|
||||||
|
|
||||||
|
@ -9,13 +9,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -26,27 +26,27 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __MEM_ALLOCATE_H_
|
#ifndef __MEM_ALLOCATE_H_
|
||||||
#define __MEM_ALLOCATE_H_
|
#define __MEM_ALLOCATE_H_
|
||||||
|
|
||||||
#include <malloc.h>
|
#include <malloc.h>
|
||||||
|
|
||||||
static inline void* _FAT_mem_allocate ( size_t size )
|
static inline void* _FAT_mem_allocate(size_t size)
|
||||||
{
|
{
|
||||||
return malloc ( size );
|
return malloc(size);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void* _FAT_mem_align ( size_t size )
|
static inline void* _FAT_mem_align(size_t size)
|
||||||
{
|
{
|
||||||
|
|
||||||
return memalign ( 32, size );
|
return memalign(32, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void _FAT_mem_free ( void* mem )
|
static inline void _FAT_mem_free(void* mem)
|
||||||
{
|
{
|
||||||
free ( mem );
|
free(mem);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif // _MEM_ALLOCATE_H
|
#endif // _MEM_ALLOCATE_H
|
||||||
|
@ -8,13 +8,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -25,7 +25,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "partition.h"
|
#include "partition.h"
|
||||||
#include "bit_ops.h"
|
#include "bit_ops.h"
|
||||||
@ -41,19 +41,18 @@
|
|||||||
sec_t _FAT_startSector;
|
sec_t _FAT_startSector;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
This device name, as known by devkitPro toolchains
|
This device name, as known by devkitPro toolchains
|
||||||
*/
|
*/
|
||||||
const char* DEVICE_NAME = "fat";
|
const char* DEVICE_NAME = "fat";
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Data offsets
|
Data offsets
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// BIOS Parameter Block offsets
|
// BIOS Parameter Block offsets
|
||||||
enum BPB
|
enum BPB
|
||||||
{
|
{
|
||||||
BPB_jmpBoot = 0x00,
|
BPB_jmpBoot = 0x00, BPB_OEMName = 0x03,
|
||||||
BPB_OEMName = 0x03,
|
|
||||||
// BIOS Parameter Block
|
// BIOS Parameter Block
|
||||||
BPB_bytesPerSector = 0x0B,
|
BPB_bytesPerSector = 0x0B,
|
||||||
BPB_sectorsPerCluster = 0x0D,
|
BPB_sectorsPerCluster = 0x0D,
|
||||||
@ -96,67 +95,66 @@ enum BPB
|
|||||||
BPB_bootSig_AA = 0x1FF
|
BPB_bootSig_AA = 0x1FF
|
||||||
};
|
};
|
||||||
|
|
||||||
static const char FAT_SIG[3] = {'F', 'A', 'T'};
|
static const char FAT_SIG[3] = { 'F', 'A', 'T' };
|
||||||
|
|
||||||
|
sec_t FindFirstValidPartition(const DISC_INTERFACE* disc)
|
||||||
sec_t FindFirstValidPartition( const DISC_INTERFACE* disc )
|
|
||||||
{
|
{
|
||||||
uint8_t part_table[16*4];
|
uint8_t part_table[16 * 4];
|
||||||
uint8_t *ptr;
|
uint8_t *ptr;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
uint8_t sectorBuffer[BYTES_PER_READ] = {0};
|
uint8_t sectorBuffer[BYTES_PER_READ] = { 0 };
|
||||||
|
|
||||||
// Read first sector of disc
|
// Read first sector of disc
|
||||||
if ( !_FAT_disc_readSectors ( disc, 0, 1, sectorBuffer ) )
|
if (!_FAT_disc_readSectors(disc, 0, 1, sectorBuffer))
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy( part_table, sectorBuffer + 0x1BE, 16*4 );
|
memcpy(part_table, sectorBuffer + 0x1BE, 16 * 4);
|
||||||
ptr = part_table;
|
ptr = part_table;
|
||||||
|
|
||||||
for ( i = 0; i < 4; i++, ptr += 16 )
|
for (i = 0; i < 4; i++, ptr += 16)
|
||||||
{
|
{
|
||||||
sec_t part_lba = u8array_to_u32( ptr, 0x8 );
|
sec_t part_lba = u8array_to_u32(ptr, 0x8);
|
||||||
|
|
||||||
if ( !memcmp( sectorBuffer + BPB_FAT16_fileSysType, FAT_SIG, sizeof( FAT_SIG ) ) ||
|
if (!memcmp(sectorBuffer + BPB_FAT16_fileSysType, FAT_SIG, sizeof(FAT_SIG)) || !memcmp(sectorBuffer
|
||||||
!memcmp( sectorBuffer + BPB_FAT32_fileSysType, FAT_SIG, sizeof( FAT_SIG ) ) )
|
+ BPB_FAT32_fileSysType, FAT_SIG, sizeof(FAT_SIG)))
|
||||||
{
|
{
|
||||||
return part_lba;
|
return part_lba;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( ptr[4] == 0 ) continue;
|
if (ptr[4] == 0) continue;
|
||||||
|
|
||||||
if ( ptr[4] == 0x0F )
|
if (ptr[4] == 0x0F)
|
||||||
{
|
{
|
||||||
sec_t part_lba2 = part_lba;
|
sec_t part_lba2 = part_lba;
|
||||||
sec_t next_lba2 = 0;
|
sec_t next_lba2 = 0;
|
||||||
int n;
|
int n;
|
||||||
|
|
||||||
for ( n = 0; n < 8; n++ ) // max 8 logic partitions
|
for (n = 0; n < 8; n++) // max 8 logic partitions
|
||||||
{
|
{
|
||||||
if ( !_FAT_disc_readSectors ( disc, part_lba + next_lba2, 1, sectorBuffer ) ) return 0;
|
if (!_FAT_disc_readSectors(disc, part_lba + next_lba2, 1, sectorBuffer)) return 0;
|
||||||
|
|
||||||
part_lba2 = part_lba + next_lba2 + u8array_to_u32( sectorBuffer, 0x1C6 ) ;
|
part_lba2 = part_lba + next_lba2 + u8array_to_u32(sectorBuffer, 0x1C6);
|
||||||
next_lba2 = u8array_to_u32( sectorBuffer, 0x1D6 );
|
next_lba2 = u8array_to_u32(sectorBuffer, 0x1D6);
|
||||||
|
|
||||||
if ( !_FAT_disc_readSectors ( disc, part_lba2, 1, sectorBuffer ) ) return 0;
|
if (!_FAT_disc_readSectors(disc, part_lba2, 1, sectorBuffer)) return 0;
|
||||||
|
|
||||||
if ( !memcmp( sectorBuffer + BPB_FAT16_fileSysType, FAT_SIG, sizeof( FAT_SIG ) ) ||
|
if (!memcmp(sectorBuffer + BPB_FAT16_fileSysType, FAT_SIG, sizeof(FAT_SIG)) || !memcmp(sectorBuffer
|
||||||
!memcmp( sectorBuffer + BPB_FAT32_fileSysType, FAT_SIG, sizeof( FAT_SIG ) ) )
|
+ BPB_FAT32_fileSysType, FAT_SIG, sizeof(FAT_SIG)))
|
||||||
{
|
{
|
||||||
return part_lba2;
|
return part_lba2;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( next_lba2 == 0 ) break;
|
if (next_lba2 == 0) break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if ( !_FAT_disc_readSectors ( disc, part_lba, 1, sectorBuffer ) ) return 0;
|
if (!_FAT_disc_readSectors(disc, part_lba, 1, sectorBuffer)) return 0;
|
||||||
if ( !memcmp( sectorBuffer + BPB_FAT16_fileSysType, FAT_SIG, sizeof( FAT_SIG ) ) ||
|
if (!memcmp(sectorBuffer + BPB_FAT16_fileSysType, FAT_SIG, sizeof(FAT_SIG)) || !memcmp(sectorBuffer
|
||||||
!memcmp( sectorBuffer + BPB_FAT32_fileSysType, FAT_SIG, sizeof( FAT_SIG ) ) )
|
+ BPB_FAT32_fileSysType, FAT_SIG, sizeof(FAT_SIG)))
|
||||||
{
|
{
|
||||||
return part_lba;
|
return part_lba;
|
||||||
}
|
}
|
||||||
@ -165,62 +163,64 @@ sec_t FindFirstValidPartition( const DISC_INTERFACE* disc )
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
PARTITION* _FAT_partition_constructor ( const DISC_INTERFACE* disc, uint32_t cacheSize, uint32_t sectorsPerPage, sec_t startSector )
|
PARTITION* _FAT_partition_constructor(const DISC_INTERFACE* disc, uint32_t cacheSize, uint32_t sectorsPerPage,
|
||||||
|
sec_t startSector)
|
||||||
{
|
{
|
||||||
PARTITION* partition;
|
PARTITION* partition;
|
||||||
uint8_t sectorBuffer[BYTES_PER_READ] = {0};
|
uint8_t sectorBuffer[BYTES_PER_READ] = { 0 };
|
||||||
|
|
||||||
// Read first sector of disc
|
// Read first sector of disc
|
||||||
if ( !_FAT_disc_readSectors ( disc, startSector, 1, sectorBuffer ) )
|
if (!_FAT_disc_readSectors(disc, startSector, 1, sectorBuffer))
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make sure it is a valid MBR or boot sector
|
// Make sure it is a valid MBR or boot sector
|
||||||
if ( ( sectorBuffer[BPB_bootSig_55] != 0x55 ) || ( sectorBuffer[BPB_bootSig_AA] != 0xAA ) )
|
if ((sectorBuffer[BPB_bootSig_55] != 0x55) || (sectorBuffer[BPB_bootSig_AA] != 0xAA))
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( startSector != 0 )
|
if (startSector != 0)
|
||||||
{
|
{
|
||||||
// We're told where to start the partition, so just accept it
|
// We're told where to start the partition, so just accept it
|
||||||
}
|
}
|
||||||
else if ( !memcmp( sectorBuffer + BPB_FAT16_fileSysType, FAT_SIG, sizeof( FAT_SIG ) ) )
|
else if (!memcmp(sectorBuffer + BPB_FAT16_fileSysType, FAT_SIG, sizeof(FAT_SIG)))
|
||||||
{
|
{
|
||||||
// Check if there is a FAT string, which indicates this is a boot sector
|
// Check if there is a FAT string, which indicates this is a boot sector
|
||||||
startSector = 0;
|
startSector = 0;
|
||||||
}
|
}
|
||||||
else if ( !memcmp( sectorBuffer + BPB_FAT32_fileSysType, FAT_SIG, sizeof( FAT_SIG ) ) )
|
else if (!memcmp(sectorBuffer + BPB_FAT32_fileSysType, FAT_SIG, sizeof(FAT_SIG)))
|
||||||
{
|
{
|
||||||
// Check for FAT32
|
// Check for FAT32
|
||||||
startSector = 0;
|
startSector = 0;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
startSector = FindFirstValidPartition( disc );
|
startSector = FindFirstValidPartition(disc);
|
||||||
if ( !_FAT_disc_readSectors ( disc, startSector, 1, sectorBuffer ) )
|
if (!_FAT_disc_readSectors(disc, startSector, 1, sectorBuffer))
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now verify that this is indeed a FAT partition
|
// Now verify that this is indeed a FAT partition
|
||||||
if ( memcmp( sectorBuffer + BPB_FAT16_fileSysType, FAT_SIG, sizeof( FAT_SIG ) ) &&
|
if (memcmp(sectorBuffer + BPB_FAT16_fileSysType, FAT_SIG, sizeof(FAT_SIG)) && memcmp(sectorBuffer
|
||||||
memcmp( sectorBuffer + BPB_FAT32_fileSysType, FAT_SIG, sizeof( FAT_SIG ) ) )
|
+ BPB_FAT32_fileSysType, FAT_SIG, sizeof(FAT_SIG)))
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
// check again for the last two cases to make sure that we really have a FAT filesystem here
|
// check again for the last two cases to make sure that we really have a FAT filesystem here
|
||||||
// and won't corrupt any data
|
// and won't corrupt any data
|
||||||
if ( memcmp( sectorBuffer + BPB_FAT16_fileSysType, "FAT", 3 ) != 0 && memcmp( sectorBuffer + BPB_FAT32_fileSysType, "FAT32", 5 ) != 0 )
|
if (memcmp(sectorBuffer + BPB_FAT16_fileSysType, "FAT", 3) != 0 && memcmp(sectorBuffer + BPB_FAT32_fileSysType,
|
||||||
|
"FAT32", 5) != 0)
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
partition = ( PARTITION* ) _FAT_mem_allocate ( sizeof( PARTITION ) );
|
partition = (PARTITION*) _FAT_mem_allocate(sizeof(PARTITION));
|
||||||
if ( partition == NULL )
|
if (partition == NULL)
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
@ -228,77 +228,82 @@ PARTITION* _FAT_partition_constructor ( const DISC_INTERFACE* disc, uint32_t cac
|
|||||||
_FAT_startSector = startSector;
|
_FAT_startSector = startSector;
|
||||||
|
|
||||||
// Init the partition lock
|
// Init the partition lock
|
||||||
_FAT_lock_init( &partition->lock );
|
_FAT_lock_init(&partition->lock);
|
||||||
|
|
||||||
// Set partition's disc interface
|
// Set partition's disc interface
|
||||||
partition->disc = disc;
|
partition->disc = disc;
|
||||||
|
|
||||||
// Store required information about the file system
|
// Store required information about the file system
|
||||||
partition->fat.sectorsPerFat = u8array_to_u16( sectorBuffer, BPB_sectorsPerFAT );
|
partition->fat.sectorsPerFat = u8array_to_u16(sectorBuffer, BPB_sectorsPerFAT);
|
||||||
if ( partition->fat.sectorsPerFat == 0 )
|
if (partition->fat.sectorsPerFat == 0)
|
||||||
{
|
{
|
||||||
partition->fat.sectorsPerFat = u8array_to_u32( sectorBuffer, BPB_FAT32_sectorsPerFAT32 );
|
partition->fat.sectorsPerFat = u8array_to_u32(sectorBuffer, BPB_FAT32_sectorsPerFAT32);
|
||||||
}
|
}
|
||||||
|
|
||||||
partition->numberOfSectors = u8array_to_u16( sectorBuffer, BPB_numSectorsSmall );
|
partition->numberOfSectors = u8array_to_u16(sectorBuffer, BPB_numSectorsSmall);
|
||||||
if ( partition->numberOfSectors == 0 )
|
if (partition->numberOfSectors == 0)
|
||||||
{
|
{
|
||||||
partition->numberOfSectors = u8array_to_u32( sectorBuffer, BPB_numSectors );
|
partition->numberOfSectors = u8array_to_u32(sectorBuffer, BPB_numSectors);
|
||||||
}
|
}
|
||||||
|
|
||||||
partition->bytesPerSector = BYTES_PER_READ; // Sector size is redefined to be 512 bytes
|
partition->bytesPerSector = BYTES_PER_READ; // Sector size is redefined to be 512 bytes
|
||||||
partition->sectorsPerCluster = sectorBuffer[BPB_sectorsPerCluster] * u8array_to_u16( sectorBuffer, BPB_bytesPerSector ) / BYTES_PER_READ;
|
partition->sectorsPerCluster = sectorBuffer[BPB_sectorsPerCluster] * u8array_to_u16(sectorBuffer,
|
||||||
|
BPB_bytesPerSector) / BYTES_PER_READ;
|
||||||
partition->bytesPerCluster = partition->bytesPerSector * partition->sectorsPerCluster;
|
partition->bytesPerCluster = partition->bytesPerSector * partition->sectorsPerCluster;
|
||||||
partition->fat.fatStart = startSector + u8array_to_u16( sectorBuffer, BPB_reservedSectors );
|
partition->fat.fatStart = startSector + u8array_to_u16(sectorBuffer, BPB_reservedSectors);
|
||||||
|
|
||||||
partition->rootDirStart = partition->fat.fatStart + ( sectorBuffer[BPB_numFATs] * partition->fat.sectorsPerFat );
|
partition->rootDirStart = partition->fat.fatStart + (sectorBuffer[BPB_numFATs] * partition->fat.sectorsPerFat);
|
||||||
partition->dataStart = partition->rootDirStart +
|
partition->dataStart = partition->rootDirStart + ((u8array_to_u16(sectorBuffer, BPB_rootEntries)
|
||||||
( ( u8array_to_u16( sectorBuffer, BPB_rootEntries ) * DIR_ENTRY_DATA_SIZE ) / partition->bytesPerSector );
|
* DIR_ENTRY_DATA_SIZE) / partition->bytesPerSector);
|
||||||
|
|
||||||
partition->totalSize = ( ( uint64_t )partition->numberOfSectors - ( partition->dataStart - startSector ) ) * ( uint64_t )partition->bytesPerSector;
|
partition->totalSize = ((uint64_t) partition->numberOfSectors - (partition->dataStart - startSector))
|
||||||
|
* (uint64_t) partition->bytesPerSector;
|
||||||
|
|
||||||
// Store info about FAT
|
// Store info about FAT
|
||||||
uint32_t clusterCount = ( partition->numberOfSectors - ( uint32_t )( partition->dataStart - startSector ) ) / partition->sectorsPerCluster;
|
uint32_t clusterCount = (partition->numberOfSectors - (uint32_t) (partition->dataStart - startSector))
|
||||||
|
/ partition->sectorsPerCluster;
|
||||||
partition->fat.lastCluster = clusterCount + CLUSTER_FIRST - 1;
|
partition->fat.lastCluster = clusterCount + CLUSTER_FIRST - 1;
|
||||||
partition->fat.firstFree = CLUSTER_FIRST;
|
partition->fat.firstFree = CLUSTER_FIRST;
|
||||||
|
|
||||||
if ( clusterCount < CLUSTERS_PER_FAT12 )
|
if (clusterCount < CLUSTERS_PER_FAT12)
|
||||||
{
|
{
|
||||||
partition->filesysType = FS_FAT12; // FAT12 volume
|
partition->filesysType = FS_FAT12; // FAT12 volume
|
||||||
}
|
}
|
||||||
else if ( clusterCount < CLUSTERS_PER_FAT16 )
|
else if (clusterCount < CLUSTERS_PER_FAT16)
|
||||||
{
|
{
|
||||||
partition->filesysType = FS_FAT16; // FAT16 volume
|
partition->filesysType = FS_FAT16; // FAT16 volume
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
partition->filesysType = FS_FAT32; // FAT32 volume
|
partition->filesysType = FS_FAT32; // FAT32 volume
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( partition->filesysType != FS_FAT32 )
|
if (partition->filesysType != FS_FAT32)
|
||||||
{
|
{
|
||||||
partition->rootDirCluster = FAT16_ROOT_DIR_CLUSTER;
|
partition->rootDirCluster = FAT16_ROOT_DIR_CLUSTER;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Set up for the FAT32 way
|
// Set up for the FAT32 way
|
||||||
partition->rootDirCluster = u8array_to_u32( sectorBuffer, BPB_FAT32_rootClus );
|
partition->rootDirCluster = u8array_to_u32(sectorBuffer, BPB_FAT32_rootClus);
|
||||||
// Check if FAT mirroring is enabled
|
// Check if FAT mirroring is enabled
|
||||||
if ( !( sectorBuffer[BPB_FAT32_extFlags] & 0x80 ) )
|
if (!(sectorBuffer[BPB_FAT32_extFlags] & 0x80))
|
||||||
{
|
{
|
||||||
// Use the active FAT
|
// Use the active FAT
|
||||||
partition->fat.fatStart = partition->fat.fatStart + ( partition->fat.sectorsPerFat * ( sectorBuffer[BPB_FAT32_extFlags] & 0x0F ) );
|
partition->fat.fatStart = partition->fat.fatStart + (partition->fat.sectorsPerFat
|
||||||
|
* (sectorBuffer[BPB_FAT32_extFlags] & 0x0F));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create a cache to use
|
// Create a cache to use
|
||||||
partition->cache = _FAT_cache_constructor ( cacheSize, sectorsPerPage, partition->disc, startSector + partition->numberOfSectors );
|
partition->cache = _FAT_cache_constructor(cacheSize, sectorsPerPage, partition->disc, startSector
|
||||||
|
+ partition->numberOfSectors);
|
||||||
|
|
||||||
// Set current directory to the root
|
// Set current directory to the root
|
||||||
partition->cwdCluster = partition->rootDirCluster;
|
partition->cwdCluster = partition->rootDirCluster;
|
||||||
|
|
||||||
// Check if this disc is writable, and set the readOnly property appropriately
|
// Check if this disc is writable, and set the readOnly property appropriately
|
||||||
partition->readOnly = !( _FAT_disc_features( disc ) & FEATURE_MEDIUM_CANWRITE );
|
partition->readOnly = !(_FAT_disc_features(disc) & FEATURE_MEDIUM_CANWRITE);
|
||||||
|
|
||||||
// There are currently no open files on this partition
|
// There are currently no open files on this partition
|
||||||
partition->openFileCount = 0;
|
partition->openFileCount = 0;
|
||||||
@ -307,41 +312,41 @@ PARTITION* _FAT_partition_constructor ( const DISC_INTERFACE* disc, uint32_t cac
|
|||||||
return partition;
|
return partition;
|
||||||
}
|
}
|
||||||
|
|
||||||
void _FAT_partition_destructor ( PARTITION* partition )
|
void _FAT_partition_destructor(PARTITION* partition)
|
||||||
{
|
{
|
||||||
FILE_STRUCT* nextFile;
|
FILE_STRUCT* nextFile;
|
||||||
|
|
||||||
_FAT_lock( &partition->lock );
|
_FAT_lock(&partition->lock);
|
||||||
|
|
||||||
// Synchronize open files
|
// Synchronize open files
|
||||||
nextFile = partition->firstOpenFile;
|
nextFile = partition->firstOpenFile;
|
||||||
while ( nextFile )
|
while (nextFile)
|
||||||
{
|
{
|
||||||
_FAT_syncToDisc ( nextFile );
|
_FAT_syncToDisc(nextFile);
|
||||||
nextFile = nextFile->nextOpenFile;
|
nextFile = nextFile->nextOpenFile;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Free memory used by the cache, writing it to disc at the same time
|
// Free memory used by the cache, writing it to disc at the same time
|
||||||
_FAT_cache_destructor ( partition->cache );
|
_FAT_cache_destructor(partition->cache);
|
||||||
|
|
||||||
// Unlock the partition and destroy the lock
|
// Unlock the partition and destroy the lock
|
||||||
_FAT_unlock( &partition->lock );
|
_FAT_unlock(&partition->lock);
|
||||||
_FAT_lock_deinit( &partition->lock );
|
_FAT_lock_deinit(&partition->lock);
|
||||||
|
|
||||||
// Free memory used by the partition
|
// Free memory used by the partition
|
||||||
_FAT_mem_free ( partition );
|
_FAT_mem_free(partition);
|
||||||
}
|
}
|
||||||
|
|
||||||
PARTITION* _FAT_partition_getPartitionFromPath ( const char* path )
|
PARTITION* _FAT_partition_getPartitionFromPath(const char* path)
|
||||||
{
|
{
|
||||||
const devoptab_t *devops;
|
const devoptab_t *devops;
|
||||||
|
|
||||||
devops = GetDeviceOpTab ( path );
|
devops = GetDeviceOpTab(path);
|
||||||
|
|
||||||
if ( !devops )
|
if (!devops)
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
return ( PARTITION* )devops->deviceData;
|
return (PARTITION*) devops->deviceData;
|
||||||
}
|
}
|
||||||
|
@ -8,13 +8,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -25,7 +25,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __PARTITION_H
|
#ifndef __PARTITION_H
|
||||||
#define __PARTITION_H
|
#define __PARTITION_H
|
||||||
@ -38,53 +38,57 @@
|
|||||||
extern const char* DEVICE_NAME;
|
extern const char* DEVICE_NAME;
|
||||||
|
|
||||||
// Filesystem type
|
// Filesystem type
|
||||||
typedef enum {FS_UNKNOWN, FS_FAT12, FS_FAT16, FS_FAT32} FS_TYPE;
|
typedef enum
|
||||||
|
{
|
||||||
|
FS_UNKNOWN, FS_FAT12, FS_FAT16, FS_FAT32
|
||||||
|
} FS_TYPE;
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
sec_t fatStart;
|
sec_t fatStart;
|
||||||
uint32_t sectorsPerFat;
|
uint32_t sectorsPerFat;
|
||||||
uint32_t lastCluster;
|
uint32_t lastCluster;
|
||||||
uint32_t firstFree;
|
uint32_t firstFree;
|
||||||
} FAT;
|
} FAT;
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
const DISC_INTERFACE* disc;
|
const DISC_INTERFACE* disc;
|
||||||
CACHE* cache;
|
CACHE* cache;
|
||||||
// Info about the partition
|
// Info about the partition
|
||||||
FS_TYPE filesysType;
|
FS_TYPE filesysType;
|
||||||
uint64_t totalSize;
|
uint64_t totalSize;
|
||||||
sec_t rootDirStart;
|
sec_t rootDirStart;
|
||||||
uint32_t rootDirCluster;
|
uint32_t rootDirCluster;
|
||||||
uint32_t numberOfSectors;
|
uint32_t numberOfSectors;
|
||||||
sec_t dataStart;
|
sec_t dataStart;
|
||||||
uint32_t bytesPerSector;
|
uint32_t bytesPerSector;
|
||||||
uint32_t sectorsPerCluster;
|
uint32_t sectorsPerCluster;
|
||||||
uint32_t bytesPerCluster;
|
uint32_t bytesPerCluster;
|
||||||
FAT fat;
|
FAT fat;
|
||||||
// Values that may change after construction
|
// Values that may change after construction
|
||||||
uint32_t cwdCluster; // Current working directory cluster
|
uint32_t cwdCluster; // Current working directory cluster
|
||||||
int openFileCount;
|
int openFileCount;
|
||||||
struct _FILE_STRUCT* firstOpenFile; // The start of a linked list of files
|
struct _FILE_STRUCT* firstOpenFile; // The start of a linked list of files
|
||||||
mutex_t lock; // A lock for partition operations
|
mutex_t lock; // A lock for partition operations
|
||||||
bool readOnly; // If this is set, then do not try writing to the disc
|
bool readOnly; // If this is set, then do not try writing to the disc
|
||||||
} PARTITION;
|
} PARTITION;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Mount the supplied device and return a pointer to the struct necessary to use it
|
Mount the supplied device and return a pointer to the struct necessary to use it
|
||||||
*/
|
*/
|
||||||
PARTITION* _FAT_partition_constructor ( const DISC_INTERFACE* disc, uint32_t cacheSize, uint32_t SectorsPerPage, sec_t startSector );
|
PARTITION* _FAT_partition_constructor(const DISC_INTERFACE* disc, uint32_t cacheSize, uint32_t SectorsPerPage,
|
||||||
|
sec_t startSector);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Dismount the device and free all structures used.
|
Dismount the device and free all structures used.
|
||||||
Will also attempt to synchronise all open files to disc.
|
Will also attempt to synchronise all open files to disc.
|
||||||
*/
|
*/
|
||||||
void _FAT_partition_destructor ( PARTITION* partition );
|
void _FAT_partition_destructor(PARTITION* partition);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Return the partition specified in a path, as taken from the devoptab.
|
Return the partition specified in a path, as taken from the devoptab.
|
||||||
*/
|
*/
|
||||||
PARTITION* _FAT_partition_getPartitionFromPath ( const char* path );
|
PARTITION* _FAT_partition_getPartitionFromPath(const char* path);
|
||||||
|
|
||||||
#endif // _PARTITION_H
|
#endif // _PARTITION_H
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -56,7 +56,6 @@
|
|||||||
#define NTFS_FIND_USER(map,usid) ntfs_find_user(map,usid)
|
#define NTFS_FIND_USER(map,usid) ntfs_find_user(map,usid)
|
||||||
#define NTFS_FIND_GROUP(map,gsid) ntfs_find_group(map,gsid)
|
#define NTFS_FIND_GROUP(map,gsid) ntfs_find_group(map,gsid)
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Matching of ntfs permissions to Linux permissions
|
* Matching of ntfs permissions to Linux permissions
|
||||||
* these constants are adapted to endianness
|
* these constants are adapted to endianness
|
||||||
@ -64,7 +63,7 @@
|
|||||||
* when checking, check one is present
|
* when checking, check one is present
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* flags which are set to mean exec, write or read */
|
/* flags which are set to mean exec, write or read */
|
||||||
|
|
||||||
#define FILE_READ (FILE_READ_DATA)
|
#define FILE_READ (FILE_READ_DATA)
|
||||||
#define FILE_WRITE (FILE_WRITE_DATA | FILE_APPEND_DATA \
|
#define FILE_WRITE (FILE_WRITE_DATA | FILE_APPEND_DATA \
|
||||||
@ -75,8 +74,8 @@
|
|||||||
| READ_CONTROL | FILE_WRITE_ATTRIBUTES | FILE_WRITE_EA)
|
| READ_CONTROL | FILE_WRITE_ATTRIBUTES | FILE_WRITE_EA)
|
||||||
#define DIR_EXEC (FILE_TRAVERSE)
|
#define DIR_EXEC (FILE_TRAVERSE)
|
||||||
|
|
||||||
/* flags tested for meaning exec, write or read */
|
/* flags tested for meaning exec, write or read */
|
||||||
/* tests for write allow for interpretation of a sticky bit */
|
/* tests for write allow for interpretation of a sticky bit */
|
||||||
|
|
||||||
#define FILE_GREAD (FILE_READ_DATA | GENERIC_READ)
|
#define FILE_GREAD (FILE_READ_DATA | GENERIC_READ)
|
||||||
#define FILE_GWRITE (FILE_WRITE_DATA | FILE_APPEND_DATA | GENERIC_WRITE)
|
#define FILE_GWRITE (FILE_WRITE_DATA | FILE_APPEND_DATA | GENERIC_WRITE)
|
||||||
@ -85,19 +84,19 @@
|
|||||||
#define DIR_GWRITE (FILE_ADD_FILE | FILE_ADD_SUBDIRECTORY | GENERIC_WRITE)
|
#define DIR_GWRITE (FILE_ADD_FILE | FILE_ADD_SUBDIRECTORY | GENERIC_WRITE)
|
||||||
#define DIR_GEXEC (FILE_TRAVERSE | GENERIC_EXECUTE)
|
#define DIR_GEXEC (FILE_TRAVERSE | GENERIC_EXECUTE)
|
||||||
|
|
||||||
/* standard owner (and administrator) rights */
|
/* standard owner (and administrator) rights */
|
||||||
|
|
||||||
#define OWNER_RIGHTS (DELETE | READ_CONTROL | WRITE_DAC | WRITE_OWNER \
|
#define OWNER_RIGHTS (DELETE | READ_CONTROL | WRITE_DAC | WRITE_OWNER \
|
||||||
| SYNCHRONIZE \
|
| SYNCHRONIZE \
|
||||||
| FILE_READ_ATTRIBUTES | FILE_WRITE_ATTRIBUTES \
|
| FILE_READ_ATTRIBUTES | FILE_WRITE_ATTRIBUTES \
|
||||||
| FILE_READ_EA | FILE_WRITE_EA)
|
| FILE_READ_EA | FILE_WRITE_EA)
|
||||||
|
|
||||||
/* standard world rights */
|
/* standard world rights */
|
||||||
|
|
||||||
#define WORLD_RIGHTS (READ_CONTROL | FILE_READ_ATTRIBUTES | FILE_READ_EA \
|
#define WORLD_RIGHTS (READ_CONTROL | FILE_READ_ATTRIBUTES | FILE_READ_EA \
|
||||||
| SYNCHRONIZE)
|
| SYNCHRONIZE)
|
||||||
|
|
||||||
/* inheritance flags for files and directories */
|
/* inheritance flags for files and directories */
|
||||||
|
|
||||||
#define FILE_INHERITANCE NO_PROPAGATE_INHERIT_ACE
|
#define FILE_INHERITANCE NO_PROPAGATE_INHERIT_ACE
|
||||||
#define DIR_INHERITANCE (OBJECT_INHERIT_ACE | CONTAINER_INHERIT_ACE)
|
#define DIR_INHERITANCE (OBJECT_INHERIT_ACE | CONTAINER_INHERIT_ACE)
|
||||||
@ -122,12 +121,13 @@ typedef char BIGSID[40];
|
|||||||
* (private to this module)
|
* (private to this module)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
struct MAPLIST {
|
struct MAPLIST
|
||||||
struct MAPLIST *next;
|
{
|
||||||
char *uidstr; /* uid text from the same record */
|
struct MAPLIST *next;
|
||||||
char *gidstr; /* gid text from the same record */
|
char *uidstr; /* uid text from the same record */
|
||||||
char *sidstr; /* sid text from the same record */
|
char *gidstr; /* gid text from the same record */
|
||||||
char maptext[LINESZ + 1];
|
char *sidstr; /* sid text from the same record */
|
||||||
|
char maptext[LINESZ + 1];
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef int (*FILEREADER)(void *fileid, char *buf, size_t size, off_t pos);
|
typedef int (*FILEREADER)(void *fileid, char *buf, size_t size, off_t pos);
|
||||||
@ -150,14 +150,11 @@ BOOL ntfs_same_sid(const SID *first, const SID *second);
|
|||||||
|
|
||||||
BOOL ntfs_is_user_sid(const SID *usid);
|
BOOL ntfs_is_user_sid(const SID *usid);
|
||||||
|
|
||||||
|
|
||||||
int ntfs_sid_size(const SID * sid);
|
int ntfs_sid_size(const SID * sid);
|
||||||
unsigned int ntfs_attr_size(const char *attr);
|
unsigned int ntfs_attr_size(const char *attr);
|
||||||
|
|
||||||
const SID *ntfs_find_usid(const struct MAPPING *usermapping,
|
const SID *ntfs_find_usid(const struct MAPPING *usermapping, uid_t uid, SID *pdefsid);
|
||||||
uid_t uid, SID *pdefsid);
|
const SID *ntfs_find_gsid(const struct MAPPING *groupmapping, gid_t gid, SID *pdefsid);
|
||||||
const SID *ntfs_find_gsid(const struct MAPPING *groupmapping,
|
|
||||||
gid_t gid, SID *pdefsid);
|
|
||||||
uid_t ntfs_find_user(const struct MAPPING *usermapping, const SID *usid);
|
uid_t ntfs_find_user(const struct MAPPING *usermapping, const SID *usid);
|
||||||
gid_t ntfs_find_group(const struct MAPPING *groupmapping, const SID * gsid);
|
gid_t ntfs_find_group(const struct MAPPING *groupmapping, const SID * gsid);
|
||||||
const SID *ntfs_acl_owner(const char *secattr);
|
const SID *ntfs_acl_owner(const char *secattr);
|
||||||
@ -168,28 +165,25 @@ BOOL ntfs_valid_posix(const struct POSIX_SECURITY *pxdesc);
|
|||||||
void ntfs_sort_posix(struct POSIX_SECURITY *pxdesc);
|
void ntfs_sort_posix(struct POSIX_SECURITY *pxdesc);
|
||||||
int ntfs_merge_mode_posix(struct POSIX_SECURITY *pxdesc, mode_t mode);
|
int ntfs_merge_mode_posix(struct POSIX_SECURITY *pxdesc, mode_t mode);
|
||||||
struct POSIX_SECURITY *ntfs_build_inherited_posix(
|
struct POSIX_SECURITY *ntfs_build_inherited_posix(
|
||||||
const struct POSIX_SECURITY *pxdesc, mode_t mode,
|
const struct POSIX_SECURITY *pxdesc, mode_t mode,
|
||||||
mode_t umask, BOOL isdir);
|
mode_t umask, BOOL isdir);
|
||||||
struct POSIX_SECURITY *ntfs_replace_acl(const struct POSIX_SECURITY *oldpxdesc,
|
struct POSIX_SECURITY *ntfs_replace_acl(const struct POSIX_SECURITY *oldpxdesc,
|
||||||
const struct POSIX_ACL *newacl, int count, BOOL deflt);
|
const struct POSIX_ACL *newacl, int count, BOOL deflt);
|
||||||
struct POSIX_SECURITY *ntfs_build_permissions_posix(
|
struct POSIX_SECURITY *ntfs_build_permissions_posix(
|
||||||
struct MAPPING* const mapping[],
|
struct MAPPING* const mapping[],
|
||||||
const char *securattr,
|
const char *securattr,
|
||||||
const SID *usid, const SID *gsid, BOOL isdir);
|
const SID *usid, const SID *gsid, BOOL isdir);
|
||||||
struct POSIX_SECURITY *ntfs_merge_descr_posix(const struct POSIX_SECURITY *first,
|
struct POSIX_SECURITY *ntfs_merge_descr_posix(const struct POSIX_SECURITY *first,
|
||||||
const struct POSIX_SECURITY *second);
|
const struct POSIX_SECURITY *second);
|
||||||
char *ntfs_build_descr_posix(struct MAPPING* const mapping[],
|
char *ntfs_build_descr_posix(struct MAPPING* const mapping[],
|
||||||
struct POSIX_SECURITY *pxdesc,
|
struct POSIX_SECURITY *pxdesc,
|
||||||
int isdir, const SID *usid, const SID *gsid);
|
int isdir, const SID *usid, const SID *gsid);
|
||||||
|
|
||||||
#endif /* POSIXACLS */
|
#endif /* POSIXACLS */
|
||||||
|
|
||||||
int ntfs_inherit_acl(const ACL *oldacl, ACL *newacl,
|
int ntfs_inherit_acl(const ACL *oldacl, ACL *newacl, const SID *usid, const SID *gsid, BOOL fordir);
|
||||||
const SID *usid, const SID *gsid, BOOL fordir);
|
int ntfs_build_permissions(const char *securattr, const SID *usid, const SID *gsid, BOOL isdir);
|
||||||
int ntfs_build_permissions(const char *securattr,
|
char *ntfs_build_descr(mode_t mode, int isdir, const SID * usid, const SID * gsid);
|
||||||
const SID *usid, const SID *gsid, BOOL isdir);
|
|
||||||
char *ntfs_build_descr(mode_t mode,
|
|
||||||
int isdir, const SID * usid, const SID * gsid);
|
|
||||||
struct MAPLIST *ntfs_read_mapping(FILEREADER reader, void *fileid);
|
struct MAPLIST *ntfs_read_mapping(FILEREADER reader, void *fileid);
|
||||||
struct MAPPING *ntfs_do_user_mapping(struct MAPLIST *firstitem);
|
struct MAPPING *ntfs_do_user_mapping(struct MAPLIST *firstitem);
|
||||||
struct MAPPING *ntfs_do_group_mapping(struct MAPLIST *firstitem);
|
struct MAPPING *ntfs_do_group_mapping(struct MAPLIST *firstitem);
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -49,12 +49,10 @@ extern ntfschar TXF_DATA[10];
|
|||||||
*
|
*
|
||||||
* TODO: Describe them.
|
* TODO: Describe them.
|
||||||
*/
|
*/
|
||||||
typedef enum {
|
typedef enum
|
||||||
LCN_HOLE = -1, /* Keep this as highest value or die! */
|
{
|
||||||
LCN_RL_NOT_MAPPED = -2,
|
LCN_HOLE = -1, /* Keep this as highest value or die! */
|
||||||
LCN_ENOENT = -3,
|
LCN_RL_NOT_MAPPED = -2, LCN_ENOENT = -3, LCN_EINVAL = -4, LCN_EIO = -5,
|
||||||
LCN_EINVAL = -4,
|
|
||||||
LCN_EIO = -5,
|
|
||||||
} ntfs_lcn_special_values;
|
} ntfs_lcn_special_values;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -75,31 +73,28 @@ typedef enum {
|
|||||||
* any modification of the search context, to automagically get the next
|
* any modification of the search context, to automagically get the next
|
||||||
* matching attribute.
|
* matching attribute.
|
||||||
*/
|
*/
|
||||||
struct _ntfs_attr_search_ctx {
|
struct _ntfs_attr_search_ctx
|
||||||
MFT_RECORD *mrec;
|
{
|
||||||
ATTR_RECORD *attr;
|
MFT_RECORD *mrec;
|
||||||
BOOL is_first;
|
ATTR_RECORD *attr;
|
||||||
ntfs_inode *ntfs_ino;
|
BOOL is_first;
|
||||||
ATTR_LIST_ENTRY *al_entry;
|
ntfs_inode *ntfs_ino;
|
||||||
ntfs_inode *base_ntfs_ino;
|
ATTR_LIST_ENTRY *al_entry;
|
||||||
MFT_RECORD *base_mrec;
|
ntfs_inode *base_ntfs_ino;
|
||||||
ATTR_RECORD *base_attr;
|
MFT_RECORD *base_mrec;
|
||||||
|
ATTR_RECORD *base_attr;
|
||||||
};
|
};
|
||||||
|
|
||||||
extern void ntfs_attr_reinit_search_ctx(ntfs_attr_search_ctx *ctx);
|
extern void ntfs_attr_reinit_search_ctx(ntfs_attr_search_ctx *ctx);
|
||||||
extern ntfs_attr_search_ctx *ntfs_attr_get_search_ctx(ntfs_inode *ni,
|
extern ntfs_attr_search_ctx *ntfs_attr_get_search_ctx(ntfs_inode *ni, MFT_RECORD *mrec);
|
||||||
MFT_RECORD *mrec);
|
|
||||||
extern void ntfs_attr_put_search_ctx(ntfs_attr_search_ctx *ctx);
|
extern void ntfs_attr_put_search_ctx(ntfs_attr_search_ctx *ctx);
|
||||||
|
|
||||||
extern int ntfs_attr_lookup(const ATTR_TYPES type, const ntfschar *name,
|
extern int ntfs_attr_lookup(const ATTR_TYPES type, const ntfschar *name, const u32 name_len, const IGNORE_CASE_BOOL ic,
|
||||||
const u32 name_len, const IGNORE_CASE_BOOL ic,
|
const VCN lowest_vcn, const u8 *val, const u32 val_len, ntfs_attr_search_ctx *ctx);
|
||||||
const VCN lowest_vcn, const u8 *val, const u32 val_len,
|
|
||||||
ntfs_attr_search_ctx *ctx);
|
|
||||||
|
|
||||||
extern int ntfs_attr_position(const ATTR_TYPES type, ntfs_attr_search_ctx *ctx);
|
extern int ntfs_attr_position(const ATTR_TYPES type, ntfs_attr_search_ctx *ctx);
|
||||||
|
|
||||||
extern ATTR_DEF *ntfs_attr_find_in_attrdef(const ntfs_volume *vol,
|
extern ATTR_DEF *ntfs_attr_find_in_attrdef(const ntfs_volume *vol, const ATTR_TYPES type);
|
||||||
const ATTR_TYPES type);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ntfs_attrs_walk - syntactic sugar for walking all attributes in an inode
|
* ntfs_attrs_walk - syntactic sugar for walking all attributes in an inode
|
||||||
@ -128,8 +123,7 @@ extern ATTR_DEF *ntfs_attr_find_in_attrdef(const ntfs_volume *vol,
|
|||||||
*/
|
*/
|
||||||
static __inline__ int ntfs_attrs_walk(ntfs_attr_search_ctx *ctx)
|
static __inline__ int ntfs_attrs_walk(ntfs_attr_search_ctx *ctx)
|
||||||
{
|
{
|
||||||
return ntfs_attr_lookup(AT_UNUSED, NULL, 0, CASE_SENSITIVE, 0,
|
return ntfs_attr_lookup(AT_UNUSED, NULL, 0, CASE_SENSITIVE, 0, NULL, 0, ctx);
|
||||||
NULL, 0, ctx);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -174,34 +168,37 @@ static __inline__ int ntfs_attrs_walk(ntfs_attr_search_ctx *ctx)
|
|||||||
* @state contains NTFS attribute specific flags describing this attribute
|
* @state contains NTFS attribute specific flags describing this attribute
|
||||||
* structure. See ntfs_attr_state_bits above.
|
* structure. See ntfs_attr_state_bits above.
|
||||||
*/
|
*/
|
||||||
struct _ntfs_attr {
|
struct _ntfs_attr
|
||||||
runlist_element *rl;
|
{
|
||||||
ntfs_inode *ni;
|
runlist_element *rl;
|
||||||
ATTR_TYPES type;
|
ntfs_inode *ni;
|
||||||
ATTR_FLAGS data_flags;
|
ATTR_TYPES type;
|
||||||
ntfschar *name;
|
ATTR_FLAGS data_flags;
|
||||||
u32 name_len;
|
ntfschar *name;
|
||||||
unsigned long state;
|
u32 name_len;
|
||||||
s64 allocated_size;
|
unsigned long state;
|
||||||
s64 data_size;
|
s64 allocated_size;
|
||||||
s64 initialized_size;
|
s64 data_size;
|
||||||
s64 compressed_size;
|
s64 initialized_size;
|
||||||
u32 compression_block_size;
|
s64 compressed_size;
|
||||||
u8 compression_block_size_bits;
|
u32 compression_block_size;
|
||||||
u8 compression_block_clusters;
|
u8 compression_block_size_bits;
|
||||||
s8 unused_runs; /* pre-reserved entries available */
|
u8 compression_block_clusters;
|
||||||
|
s8 unused_runs; /* pre-reserved entries available */
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* enum ntfs_attr_state_bits - bits for the state field in the ntfs_attr
|
* enum ntfs_attr_state_bits - bits for the state field in the ntfs_attr
|
||||||
* structure
|
* structure
|
||||||
*/
|
*/
|
||||||
typedef enum {
|
typedef enum
|
||||||
NA_Initialized, /* 1: structure is initialized. */
|
{
|
||||||
NA_NonResident, /* 1: Attribute is not resident. */
|
NA_Initialized, /* 1: structure is initialized. */
|
||||||
NA_BeingNonResident, /* 1: Attribute is being made not resident. */
|
NA_NonResident, /* 1: Attribute is not resident. */
|
||||||
NA_FullyMapped, /* 1: Attribute has been fully mapped */
|
NA_BeingNonResident, /* 1: Attribute is being made not resident. */
|
||||||
NA_ComprClosing, /* 1: Compressed attribute is being closed */
|
NA_FullyMapped, /* 1: Attribute has been fully mapped */
|
||||||
|
NA_ComprClosing,
|
||||||
|
/* 1: Compressed attribute is being closed */
|
||||||
} ntfs_attr_state_bits;
|
} ntfs_attr_state_bits;
|
||||||
|
|
||||||
#define test_nattr_flag(na, flag) test_bit(NA_##flag, (na)->state)
|
#define test_nattr_flag(na, flag) test_bit(NA_##flag, (na)->state)
|
||||||
@ -234,8 +231,8 @@ extern void NAttrSet##func_name(ntfs_attr *na); \
|
|||||||
extern void NAttrClear##func_name(ntfs_attr *na);
|
extern void NAttrClear##func_name(ntfs_attr *na);
|
||||||
|
|
||||||
GenNAttrIno(Compressed, FILE_ATTR_COMPRESSED)
|
GenNAttrIno(Compressed, FILE_ATTR_COMPRESSED)
|
||||||
GenNAttrIno(Encrypted, FILE_ATTR_ENCRYPTED)
|
GenNAttrIno(Encrypted, FILE_ATTR_ENCRYPTED)
|
||||||
GenNAttrIno(Sparse, FILE_ATTR_SPARSE_FILE)
|
GenNAttrIno(Sparse, FILE_ATTR_SPARSE_FILE)
|
||||||
#undef GenNAttrIno
|
#undef GenNAttrIno
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -243,54 +240,46 @@ GenNAttrIno(Sparse, FILE_ATTR_SPARSE_FILE)
|
|||||||
*
|
*
|
||||||
* For convenience. Used in the attr structure.
|
* For convenience. Used in the attr structure.
|
||||||
*/
|
*/
|
||||||
typedef union {
|
typedef union
|
||||||
u8 _default; /* Unnamed u8 to serve as default when just using
|
{
|
||||||
a_val without specifying any of the below. */
|
u8 _default; /* Unnamed u8 to serve as default when just using
|
||||||
STANDARD_INFORMATION std_inf;
|
a_val without specifying any of the below. */
|
||||||
ATTR_LIST_ENTRY al_entry;
|
STANDARD_INFORMATION std_inf;
|
||||||
FILE_NAME_ATTR filename;
|
ATTR_LIST_ENTRY al_entry;
|
||||||
OBJECT_ID_ATTR obj_id;
|
FILE_NAME_ATTR filename;
|
||||||
SECURITY_DESCRIPTOR_ATTR sec_desc;
|
OBJECT_ID_ATTR obj_id;
|
||||||
VOLUME_NAME vol_name;
|
SECURITY_DESCRIPTOR_ATTR sec_desc;
|
||||||
VOLUME_INFORMATION vol_inf;
|
VOLUME_NAME vol_name;
|
||||||
DATA_ATTR data;
|
VOLUME_INFORMATION vol_inf;
|
||||||
INDEX_ROOT index_root;
|
DATA_ATTR data;
|
||||||
INDEX_BLOCK index_blk;
|
INDEX_ROOT index_root;
|
||||||
BITMAP_ATTR bmp;
|
INDEX_BLOCK index_blk;
|
||||||
REPARSE_POINT reparse;
|
BITMAP_ATTR bmp;
|
||||||
EA_INFORMATION ea_inf;
|
REPARSE_POINT reparse;
|
||||||
EA_ATTR ea;
|
EA_INFORMATION ea_inf;
|
||||||
PROPERTY_SET property_set;
|
EA_ATTR ea;
|
||||||
LOGGED_UTILITY_STREAM logged_util_stream;
|
PROPERTY_SET property_set;
|
||||||
EFS_ATTR_HEADER efs;
|
LOGGED_UTILITY_STREAM logged_util_stream;
|
||||||
|
EFS_ATTR_HEADER efs;
|
||||||
} attr_val;
|
} attr_val;
|
||||||
|
|
||||||
extern void ntfs_attr_init(ntfs_attr *na, const BOOL non_resident,
|
extern void ntfs_attr_init(ntfs_attr *na, const BOOL non_resident, const ATTR_FLAGS data_flags, const BOOL encrypted,
|
||||||
const ATTR_FLAGS data_flags, const BOOL encrypted,
|
const BOOL sparse, const s64 allocated_size, const s64 data_size, const s64 initialized_size,
|
||||||
const BOOL sparse,
|
const s64 compressed_size, const u8 compression_unit);
|
||||||
const s64 allocated_size, const s64 data_size,
|
|
||||||
const s64 initialized_size, const s64 compressed_size,
|
|
||||||
const u8 compression_unit);
|
|
||||||
|
|
||||||
/* warning : in the following "name" has to be freeable */
|
/* warning : in the following "name" has to be freeable */
|
||||||
/* or one of constants AT_UNNAMED, NTFS_INDEX_I30 or STREAM_SDS */
|
/* or one of constants AT_UNNAMED, NTFS_INDEX_I30 or STREAM_SDS */
|
||||||
extern ntfs_attr *ntfs_attr_open(ntfs_inode *ni, const ATTR_TYPES type,
|
extern ntfs_attr *ntfs_attr_open(ntfs_inode *ni, const ATTR_TYPES type, ntfschar *name, u32 name_len);
|
||||||
ntfschar *name, u32 name_len);
|
|
||||||
extern void ntfs_attr_close(ntfs_attr *na);
|
extern void ntfs_attr_close(ntfs_attr *na);
|
||||||
|
|
||||||
extern s64 ntfs_attr_pread(ntfs_attr *na, const s64 pos, s64 count,
|
extern s64 ntfs_attr_pread(ntfs_attr *na, const s64 pos, s64 count, void *b);
|
||||||
void *b);
|
extern s64 ntfs_attr_pwrite(ntfs_attr *na, const s64 pos, s64 count, const void *b);
|
||||||
extern s64 ntfs_attr_pwrite(ntfs_attr *na, const s64 pos, s64 count,
|
|
||||||
const void *b);
|
|
||||||
extern int ntfs_attr_pclose(ntfs_attr *na);
|
extern int ntfs_attr_pclose(ntfs_attr *na);
|
||||||
|
|
||||||
extern void *ntfs_attr_readall(ntfs_inode *ni, const ATTR_TYPES type,
|
extern void *ntfs_attr_readall(ntfs_inode *ni, const ATTR_TYPES type, ntfschar *name, u32 name_len, s64 *data_size);
|
||||||
ntfschar *name, u32 name_len, s64 *data_size);
|
|
||||||
|
|
||||||
extern s64 ntfs_attr_mst_pread(ntfs_attr *na, const s64 pos,
|
extern s64 ntfs_attr_mst_pread(ntfs_attr *na, const s64 pos, const s64 bk_cnt, const u32 bk_size, void *dst);
|
||||||
const s64 bk_cnt, const u32 bk_size, void *dst);
|
extern s64 ntfs_attr_mst_pwrite(ntfs_attr *na, const s64 pos, s64 bk_cnt, const u32 bk_size, void *src);
|
||||||
extern s64 ntfs_attr_mst_pwrite(ntfs_attr *na, const s64 pos,
|
|
||||||
s64 bk_cnt, const u32 bk_size, void *src);
|
|
||||||
|
|
||||||
extern int ntfs_attr_map_runlist(ntfs_attr *na, VCN vcn);
|
extern int ntfs_attr_map_runlist(ntfs_attr *na, VCN vcn);
|
||||||
extern int ntfs_attr_map_whole_runlist(ntfs_attr *na);
|
extern int ntfs_attr_map_whole_runlist(ntfs_attr *na);
|
||||||
@ -298,33 +287,26 @@ extern int ntfs_attr_map_whole_runlist(ntfs_attr *na);
|
|||||||
extern LCN ntfs_attr_vcn_to_lcn(ntfs_attr *na, const VCN vcn);
|
extern LCN ntfs_attr_vcn_to_lcn(ntfs_attr *na, const VCN vcn);
|
||||||
extern runlist_element *ntfs_attr_find_vcn(ntfs_attr *na, const VCN vcn);
|
extern runlist_element *ntfs_attr_find_vcn(ntfs_attr *na, const VCN vcn);
|
||||||
|
|
||||||
extern int ntfs_attr_size_bounds_check(const ntfs_volume *vol,
|
extern int ntfs_attr_size_bounds_check(const ntfs_volume *vol, const ATTR_TYPES type, const s64 size);
|
||||||
const ATTR_TYPES type, const s64 size);
|
extern int ntfs_attr_can_be_resident(const ntfs_volume *vol, const ATTR_TYPES type);
|
||||||
extern int ntfs_attr_can_be_resident(const ntfs_volume *vol,
|
int ntfs_attr_make_non_resident(ntfs_attr *na, ntfs_attr_search_ctx *ctx);
|
||||||
const ATTR_TYPES type);
|
|
||||||
int ntfs_attr_make_non_resident(ntfs_attr *na,
|
|
||||||
ntfs_attr_search_ctx *ctx);
|
|
||||||
int ntfs_attr_force_non_resident(ntfs_attr *na);
|
int ntfs_attr_force_non_resident(ntfs_attr *na);
|
||||||
extern int ntfs_make_room_for_attr(MFT_RECORD *m, u8 *pos, u32 size);
|
extern int ntfs_make_room_for_attr(MFT_RECORD *m, u8 *pos, u32 size);
|
||||||
|
|
||||||
extern int ntfs_resident_attr_record_add(ntfs_inode *ni, ATTR_TYPES type,
|
extern int ntfs_resident_attr_record_add(ntfs_inode *ni, ATTR_TYPES type, ntfschar *name, u8 name_len, u8 *val,
|
||||||
ntfschar *name, u8 name_len, u8 *val, u32 size,
|
u32 size, ATTR_FLAGS flags);
|
||||||
ATTR_FLAGS flags);
|
extern int ntfs_non_resident_attr_record_add(ntfs_inode *ni, ATTR_TYPES type, ntfschar *name, u8 name_len,
|
||||||
extern int ntfs_non_resident_attr_record_add(ntfs_inode *ni, ATTR_TYPES type,
|
VCN lowest_vcn, int dataruns_size, ATTR_FLAGS flags);
|
||||||
ntfschar *name, u8 name_len, VCN lowest_vcn, int dataruns_size,
|
|
||||||
ATTR_FLAGS flags);
|
|
||||||
extern int ntfs_attr_record_rm(ntfs_attr_search_ctx *ctx);
|
extern int ntfs_attr_record_rm(ntfs_attr_search_ctx *ctx);
|
||||||
|
|
||||||
extern int ntfs_attr_add(ntfs_inode *ni, ATTR_TYPES type,
|
extern int ntfs_attr_add(ntfs_inode *ni, ATTR_TYPES type, ntfschar *name, u8 name_len, u8 *val, s64 size);
|
||||||
ntfschar *name, u8 name_len, u8 *val, s64 size);
|
extern int ntfs_attr_set_flags(ntfs_inode *ni, ATTR_TYPES type, ntfschar *name, u8 name_len, ATTR_FLAGS flags,
|
||||||
extern int ntfs_attr_set_flags(ntfs_inode *ni, ATTR_TYPES type,
|
ATTR_FLAGS mask);
|
||||||
ntfschar *name, u8 name_len, ATTR_FLAGS flags, ATTR_FLAGS mask);
|
|
||||||
extern int ntfs_attr_rm(ntfs_attr *na);
|
extern int ntfs_attr_rm(ntfs_attr *na);
|
||||||
|
|
||||||
extern int ntfs_attr_record_resize(MFT_RECORD *m, ATTR_RECORD *a, u32 new_size);
|
extern int ntfs_attr_record_resize(MFT_RECORD *m, ATTR_RECORD *a, u32 new_size);
|
||||||
|
|
||||||
extern int ntfs_resident_attr_value_resize(MFT_RECORD *m, ATTR_RECORD *a,
|
extern int ntfs_resident_attr_value_resize(MFT_RECORD *m, ATTR_RECORD *a, const u32 new_size);
|
||||||
const u32 new_size);
|
|
||||||
|
|
||||||
extern int ntfs_attr_record_move_to(ntfs_attr_search_ctx *ctx, ntfs_inode *ni);
|
extern int ntfs_attr_record_move_to(ntfs_attr_search_ctx *ctx, ntfs_inode *ni);
|
||||||
extern int ntfs_attr_record_move_away(ntfs_attr_search_ctx *ctx, int extra);
|
extern int ntfs_attr_record_move_away(ntfs_attr_search_ctx *ctx, int extra);
|
||||||
@ -360,16 +342,13 @@ extern s64 ntfs_get_attribute_value_length(const ATTR_RECORD *a);
|
|||||||
* then nothing was read due to a zero-length attribute value, otherwise
|
* then nothing was read due to a zero-length attribute value, otherwise
|
||||||
* errno describes the error.
|
* errno describes the error.
|
||||||
*/
|
*/
|
||||||
extern s64 ntfs_get_attribute_value(const ntfs_volume *vol,
|
extern s64 ntfs_get_attribute_value(const ntfs_volume *vol, const ATTR_RECORD *a, u8 *b);
|
||||||
const ATTR_RECORD *a, u8 *b);
|
|
||||||
|
|
||||||
extern void ntfs_attr_name_free(char **name);
|
extern void ntfs_attr_name_free(char **name);
|
||||||
extern char *ntfs_attr_name_get(const ntfschar *uname, const int uname_len);
|
extern char *ntfs_attr_name_get(const ntfschar *uname, const int uname_len);
|
||||||
extern int ntfs_attr_exist(ntfs_inode *ni, const ATTR_TYPES type,
|
extern int ntfs_attr_exist(ntfs_inode *ni, const ATTR_TYPES type, ntfschar *name, u32 name_len);
|
||||||
ntfschar *name, u32 name_len);
|
extern int ntfs_attr_remove(ntfs_inode *ni, const ATTR_TYPES type, ntfschar *name, u32 name_len);
|
||||||
extern int ntfs_attr_remove(ntfs_inode *ni, const ATTR_TYPES type,
|
extern s64 ntfs_attr_get_free_bits(ntfs_attr *na);
|
||||||
ntfschar *name, u32 name_len);
|
|
||||||
extern s64 ntfs_attr_get_free_bits(ntfs_attr *na);
|
|
||||||
|
|
||||||
#endif /* defined _NTFS_ATTRIB_H */
|
#endif /* defined _NTFS_ATTRIB_H */
|
||||||
|
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -60,36 +60,39 @@
|
|||||||
*/
|
*/
|
||||||
int ntfs_attrlist_need(ntfs_inode *ni)
|
int ntfs_attrlist_need(ntfs_inode *ni)
|
||||||
{
|
{
|
||||||
ATTR_LIST_ENTRY *ale;
|
ATTR_LIST_ENTRY *ale;
|
||||||
|
|
||||||
if (!ni) {
|
if (!ni)
|
||||||
ntfs_log_trace("Invalid arguments.\n");
|
{
|
||||||
errno = EINVAL;
|
ntfs_log_trace("Invalid arguments.\n");
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
ntfs_log_trace("Entering for inode 0x%llx.\n", (long long) ni->mft_no);
|
ntfs_log_trace("Entering for inode 0x%llx.\n", (long long) ni->mft_no);
|
||||||
|
|
||||||
if (!NInoAttrList(ni)) {
|
if (!NInoAttrList(ni))
|
||||||
ntfs_log_trace("Inode haven't got attribute list.\n");
|
{
|
||||||
errno = EINVAL;
|
ntfs_log_trace("Inode haven't got attribute list.\n");
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
if (!ni->attr_list) {
|
if (!ni->attr_list)
|
||||||
ntfs_log_trace("Corrupt in-memory struct.\n");
|
{
|
||||||
errno = EINVAL;
|
ntfs_log_trace("Corrupt in-memory struct.\n");
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
errno = 0;
|
errno = 0;
|
||||||
ale = (ATTR_LIST_ENTRY *)ni->attr_list;
|
ale = (ATTR_LIST_ENTRY *) ni->attr_list;
|
||||||
while ((u8*)ale < ni->attr_list + ni->attr_list_size) {
|
while ((u8*) ale < ni->attr_list + ni->attr_list_size)
|
||||||
if (MREF_LE(ale->mft_reference) != ni->mft_no)
|
{
|
||||||
return 1;
|
if (MREF_LE(ale->mft_reference) != ni->mft_no) return 1;
|
||||||
ale = (ATTR_LIST_ENTRY *)((u8*)ale + le16_to_cpu(ale->length));
|
ale = (ATTR_LIST_ENTRY *) ((u8*) ale + le16_to_cpu(ale->length));
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -106,131 +109,130 @@ int ntfs_attrlist_need(ntfs_inode *ni)
|
|||||||
*/
|
*/
|
||||||
int ntfs_attrlist_entry_add(ntfs_inode *ni, ATTR_RECORD *attr)
|
int ntfs_attrlist_entry_add(ntfs_inode *ni, ATTR_RECORD *attr)
|
||||||
{
|
{
|
||||||
ATTR_LIST_ENTRY *ale;
|
ATTR_LIST_ENTRY *ale;
|
||||||
MFT_REF mref;
|
MFT_REF mref;
|
||||||
ntfs_attr *na = NULL;
|
ntfs_attr *na = NULL;
|
||||||
ntfs_attr_search_ctx *ctx;
|
ntfs_attr_search_ctx *ctx;
|
||||||
u8 *new_al;
|
u8 *new_al;
|
||||||
int entry_len, entry_offset, err;
|
int entry_len, entry_offset, err;
|
||||||
|
|
||||||
ntfs_log_trace("Entering for inode 0x%llx, attr 0x%x.\n",
|
ntfs_log_trace("Entering for inode 0x%llx, attr 0x%x.\n",
|
||||||
(long long) ni->mft_no,
|
(long long) ni->mft_no,
|
||||||
(unsigned) le32_to_cpu(attr->type));
|
(unsigned) le32_to_cpu(attr->type));
|
||||||
|
|
||||||
if (!ni || !attr) {
|
if (!ni || !attr)
|
||||||
ntfs_log_trace("Invalid arguments.\n");
|
{
|
||||||
errno = EINVAL;
|
ntfs_log_trace("Invalid arguments.\n");
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
mref = MK_LE_MREF(ni->mft_no, le16_to_cpu(ni->mrec->sequence_number));
|
mref = MK_LE_MREF(ni->mft_no, le16_to_cpu(ni->mrec->sequence_number));
|
||||||
|
|
||||||
if (ni->nr_extents == -1)
|
if (ni->nr_extents == -1) ni = ni->base_ni;
|
||||||
ni = ni->base_ni;
|
|
||||||
|
|
||||||
if (!NInoAttrList(ni)) {
|
if (!NInoAttrList(ni))
|
||||||
ntfs_log_trace("Attribute list isn't present.\n");
|
{
|
||||||
errno = ENOENT;
|
ntfs_log_trace("Attribute list isn't present.\n");
|
||||||
return -1;
|
errno = ENOENT;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
/* Determine size and allocate memory for new attribute list. */
|
/* Determine size and allocate memory for new attribute list. */
|
||||||
entry_len = (sizeof(ATTR_LIST_ENTRY) + sizeof(ntfschar) *
|
entry_len = (sizeof(ATTR_LIST_ENTRY) + sizeof(ntfschar) * attr->name_length + 7) & ~7;
|
||||||
attr->name_length + 7) & ~7;
|
new_al = ntfs_calloc(ni->attr_list_size + entry_len);
|
||||||
new_al = ntfs_calloc(ni->attr_list_size + entry_len);
|
if (!new_al) return -1;
|
||||||
if (!new_al)
|
|
||||||
return -1;
|
|
||||||
|
|
||||||
/* Find place for the new entry. */
|
/* Find place for the new entry. */
|
||||||
ctx = ntfs_attr_get_search_ctx(ni, NULL);
|
ctx = ntfs_attr_get_search_ctx(ni, NULL);
|
||||||
if (!ctx) {
|
if (!ctx)
|
||||||
err = errno;
|
{
|
||||||
goto err_out;
|
err = errno;
|
||||||
}
|
goto err_out;
|
||||||
if (!ntfs_attr_lookup(attr->type, (attr->name_length) ? (ntfschar*)
|
}
|
||||||
((u8*)attr + le16_to_cpu(attr->name_offset)) :
|
if (!ntfs_attr_lookup(attr->type, (attr->name_length) ? (ntfschar*) ((u8*) attr + le16_to_cpu(attr->name_offset))
|
||||||
AT_UNNAMED, attr->name_length, CASE_SENSITIVE,
|
: AT_UNNAMED, attr->name_length, CASE_SENSITIVE, (attr->non_resident) ? le64_to_cpu(attr->lowest_vcn) : 0,
|
||||||
(attr->non_resident) ? le64_to_cpu(attr->lowest_vcn) :
|
(attr->non_resident) ? NULL : ((u8*) attr + le16_to_cpu(attr->value_offset)), (attr->non_resident) ?
|
||||||
0, (attr->non_resident) ? NULL : ((u8*)attr +
|
0
|
||||||
le16_to_cpu(attr->value_offset)), (attr->non_resident) ?
|
: le32_to_cpu(attr->value_length), ctx))
|
||||||
0 : le32_to_cpu(attr->value_length), ctx)) {
|
{
|
||||||
/* Found some extent, check it to be before new extent. */
|
/* Found some extent, check it to be before new extent. */
|
||||||
if (ctx->al_entry->lowest_vcn == attr->lowest_vcn) {
|
if (ctx->al_entry->lowest_vcn == attr->lowest_vcn)
|
||||||
err = EEXIST;
|
{
|
||||||
ntfs_log_trace("Such attribute already present in the "
|
err = EEXIST;
|
||||||
"attribute list.\n");
|
ntfs_log_trace("Such attribute already present in the "
|
||||||
ntfs_attr_put_search_ctx(ctx);
|
"attribute list.\n");
|
||||||
goto err_out;
|
ntfs_attr_put_search_ctx(ctx);
|
||||||
}
|
goto err_out;
|
||||||
/* Add new entry after this extent. */
|
}
|
||||||
ale = (ATTR_LIST_ENTRY*)((u8*)ctx->al_entry +
|
/* Add new entry after this extent. */
|
||||||
le16_to_cpu(ctx->al_entry->length));
|
ale = (ATTR_LIST_ENTRY*) ((u8*) ctx->al_entry + le16_to_cpu(ctx->al_entry->length));
|
||||||
} else {
|
}
|
||||||
/* Check for real errors. */
|
else
|
||||||
if (errno != ENOENT) {
|
{
|
||||||
err = errno;
|
/* Check for real errors. */
|
||||||
ntfs_log_trace("Attribute lookup failed.\n");
|
if (errno != ENOENT)
|
||||||
ntfs_attr_put_search_ctx(ctx);
|
{
|
||||||
goto err_out;
|
err = errno;
|
||||||
}
|
ntfs_log_trace("Attribute lookup failed.\n");
|
||||||
/* No previous extents found. */
|
ntfs_attr_put_search_ctx(ctx);
|
||||||
ale = ctx->al_entry;
|
goto err_out;
|
||||||
}
|
}
|
||||||
/* Don't need it anymore, @ctx->al_entry points to @ni->attr_list. */
|
/* No previous extents found. */
|
||||||
ntfs_attr_put_search_ctx(ctx);
|
ale = ctx->al_entry;
|
||||||
|
}
|
||||||
|
/* Don't need it anymore, @ctx->al_entry points to @ni->attr_list. */
|
||||||
|
ntfs_attr_put_search_ctx(ctx);
|
||||||
|
|
||||||
/* Determine new entry offset. */
|
/* Determine new entry offset. */
|
||||||
entry_offset = ((u8 *)ale - ni->attr_list);
|
entry_offset = ((u8 *) ale - ni->attr_list);
|
||||||
/* Set pointer to new entry. */
|
/* Set pointer to new entry. */
|
||||||
ale = (ATTR_LIST_ENTRY *)(new_al + entry_offset);
|
ale = (ATTR_LIST_ENTRY *) (new_al + entry_offset);
|
||||||
/* Zero it to fix valgrind warning. */
|
/* Zero it to fix valgrind warning. */
|
||||||
memset(ale, 0, entry_len);
|
memset(ale, 0, entry_len);
|
||||||
/* Form new entry. */
|
/* Form new entry. */
|
||||||
ale->type = attr->type;
|
ale->type = attr->type;
|
||||||
ale->length = cpu_to_le16(entry_len);
|
ale->length = cpu_to_le16(entry_len);
|
||||||
ale->name_length = attr->name_length;
|
ale->name_length = attr->name_length;
|
||||||
ale->name_offset = offsetof(ATTR_LIST_ENTRY, name);
|
ale->name_offset = offsetof(ATTR_LIST_ENTRY, name);
|
||||||
if (attr->non_resident)
|
if (attr->non_resident)
|
||||||
ale->lowest_vcn = attr->lowest_vcn;
|
ale->lowest_vcn = attr->lowest_vcn;
|
||||||
else
|
else ale->lowest_vcn = 0;
|
||||||
ale->lowest_vcn = 0;
|
ale->mft_reference = mref;
|
||||||
ale->mft_reference = mref;
|
ale->instance = attr->instance;
|
||||||
ale->instance = attr->instance;
|
memcpy(ale->name, (u8 *) attr + le16_to_cpu(attr->name_offset), attr->name_length * sizeof(ntfschar));
|
||||||
memcpy(ale->name, (u8 *)attr + le16_to_cpu(attr->name_offset),
|
|
||||||
attr->name_length * sizeof(ntfschar));
|
|
||||||
|
|
||||||
/* Resize $ATTRIBUTE_LIST to new length. */
|
/* Resize $ATTRIBUTE_LIST to new length. */
|
||||||
na = ntfs_attr_open(ni, AT_ATTRIBUTE_LIST, AT_UNNAMED, 0);
|
na = ntfs_attr_open(ni, AT_ATTRIBUTE_LIST, AT_UNNAMED, 0);
|
||||||
if (!na) {
|
if (!na)
|
||||||
err = errno;
|
{
|
||||||
ntfs_log_trace("Failed to open $ATTRIBUTE_LIST attribute.\n");
|
err = errno;
|
||||||
goto err_out;
|
ntfs_log_trace("Failed to open $ATTRIBUTE_LIST attribute.\n");
|
||||||
}
|
goto err_out;
|
||||||
if (ntfs_attr_truncate(na, ni->attr_list_size + entry_len)) {
|
}
|
||||||
err = errno;
|
if (ntfs_attr_truncate(na, ni->attr_list_size + entry_len))
|
||||||
ntfs_log_trace("$ATTRIBUTE_LIST resize failed.\n");
|
{
|
||||||
goto err_out;
|
err = errno;
|
||||||
}
|
ntfs_log_trace("$ATTRIBUTE_LIST resize failed.\n");
|
||||||
|
goto err_out;
|
||||||
|
}
|
||||||
|
|
||||||
/* Copy entries from old attribute list to new. */
|
/* Copy entries from old attribute list to new. */
|
||||||
memcpy(new_al, ni->attr_list, entry_offset);
|
memcpy(new_al, ni->attr_list, entry_offset);
|
||||||
memcpy(new_al + entry_offset + entry_len, ni->attr_list +
|
memcpy(new_al + entry_offset + entry_len, ni->attr_list + entry_offset, ni->attr_list_size - entry_offset);
|
||||||
entry_offset, ni->attr_list_size - entry_offset);
|
|
||||||
|
|
||||||
/* Set new runlist. */
|
/* Set new runlist. */
|
||||||
free(ni->attr_list);
|
free(ni->attr_list);
|
||||||
ni->attr_list = new_al;
|
ni->attr_list = new_al;
|
||||||
ni->attr_list_size = ni->attr_list_size + entry_len;
|
ni->attr_list_size = ni->attr_list_size + entry_len;
|
||||||
NInoAttrListSetDirty(ni);
|
NInoAttrListSetDirty(ni);
|
||||||
/* Done! */
|
/* Done! */
|
||||||
ntfs_attr_close(na);
|
ntfs_attr_close(na);
|
||||||
return 0;
|
return 0;
|
||||||
err_out:
|
err_out: if (na) ntfs_attr_close(na);
|
||||||
if (na)
|
free(new_al);
|
||||||
ntfs_attr_close(na);
|
errno = err;
|
||||||
free(new_al);
|
return -1;
|
||||||
errno = err;
|
|
||||||
return -1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -243,72 +245,72 @@ err_out:
|
|||||||
*/
|
*/
|
||||||
int ntfs_attrlist_entry_rm(ntfs_attr_search_ctx *ctx)
|
int ntfs_attrlist_entry_rm(ntfs_attr_search_ctx *ctx)
|
||||||
{
|
{
|
||||||
u8 *new_al;
|
u8 *new_al;
|
||||||
int new_al_len;
|
int new_al_len;
|
||||||
ntfs_inode *base_ni;
|
ntfs_inode *base_ni;
|
||||||
ntfs_attr *na;
|
ntfs_attr *na;
|
||||||
ATTR_LIST_ENTRY *ale;
|
ATTR_LIST_ENTRY *ale;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (!ctx || !ctx->ntfs_ino || !ctx->al_entry) {
|
if (!ctx || !ctx->ntfs_ino || !ctx->al_entry)
|
||||||
ntfs_log_trace("Invalid arguments.\n");
|
{
|
||||||
errno = EINVAL;
|
ntfs_log_trace("Invalid arguments.\n");
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
if (ctx->base_ntfs_ino)
|
if (ctx->base_ntfs_ino)
|
||||||
base_ni = ctx->base_ntfs_ino;
|
base_ni = ctx->base_ntfs_ino;
|
||||||
else
|
else base_ni = ctx->ntfs_ino;
|
||||||
base_ni = ctx->ntfs_ino;
|
ale = ctx->al_entry;
|
||||||
ale = ctx->al_entry;
|
|
||||||
|
|
||||||
ntfs_log_trace("Entering for inode 0x%llx, attr 0x%x, lowest_vcn %lld.\n",
|
ntfs_log_trace("Entering for inode 0x%llx, attr 0x%x, lowest_vcn %lld.\n",
|
||||||
(long long) ctx->ntfs_ino->mft_no,
|
(long long) ctx->ntfs_ino->mft_no,
|
||||||
(unsigned) le32_to_cpu(ctx->al_entry->type),
|
(unsigned) le32_to_cpu(ctx->al_entry->type),
|
||||||
(long long) le64_to_cpu(ctx->al_entry->lowest_vcn));
|
(long long) le64_to_cpu(ctx->al_entry->lowest_vcn));
|
||||||
|
|
||||||
if (!NInoAttrList(base_ni)) {
|
if (!NInoAttrList(base_ni))
|
||||||
ntfs_log_trace("Attribute list isn't present.\n");
|
{
|
||||||
errno = ENOENT;
|
ntfs_log_trace("Attribute list isn't present.\n");
|
||||||
return -1;
|
errno = ENOENT;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
/* Allocate memory for new attribute list. */
|
/* Allocate memory for new attribute list. */
|
||||||
new_al_len = base_ni->attr_list_size - le16_to_cpu(ale->length);
|
new_al_len = base_ni->attr_list_size - le16_to_cpu(ale->length);
|
||||||
new_al = ntfs_calloc(new_al_len);
|
new_al = ntfs_calloc(new_al_len);
|
||||||
if (!new_al)
|
if (!new_al) return -1;
|
||||||
return -1;
|
|
||||||
|
|
||||||
/* Reisze $ATTRIBUTE_LIST to new length. */
|
/* Reisze $ATTRIBUTE_LIST to new length. */
|
||||||
na = ntfs_attr_open(base_ni, AT_ATTRIBUTE_LIST, AT_UNNAMED, 0);
|
na = ntfs_attr_open(base_ni, AT_ATTRIBUTE_LIST, AT_UNNAMED, 0);
|
||||||
if (!na) {
|
if (!na)
|
||||||
err = errno;
|
{
|
||||||
ntfs_log_trace("Failed to open $ATTRIBUTE_LIST attribute.\n");
|
err = errno;
|
||||||
goto err_out;
|
ntfs_log_trace("Failed to open $ATTRIBUTE_LIST attribute.\n");
|
||||||
}
|
goto err_out;
|
||||||
if (ntfs_attr_truncate(na, new_al_len)) {
|
}
|
||||||
err = errno;
|
if (ntfs_attr_truncate(na, new_al_len))
|
||||||
ntfs_log_trace("$ATTRIBUTE_LIST resize failed.\n");
|
{
|
||||||
goto err_out;
|
err = errno;
|
||||||
}
|
ntfs_log_trace("$ATTRIBUTE_LIST resize failed.\n");
|
||||||
|
goto err_out;
|
||||||
|
}
|
||||||
|
|
||||||
/* Copy entries from old attribute list to new. */
|
/* Copy entries from old attribute list to new. */
|
||||||
memcpy(new_al, base_ni->attr_list, (u8*)ale - base_ni->attr_list);
|
memcpy(new_al, base_ni->attr_list, (u8*) ale - base_ni->attr_list);
|
||||||
memcpy(new_al + ((u8*)ale - base_ni->attr_list), (u8*)ale + le16_to_cpu(
|
memcpy(new_al + ((u8*) ale - base_ni->attr_list), (u8*) ale + le16_to_cpu(
|
||||||
ale->length), new_al_len - ((u8*)ale - base_ni->attr_list));
|
ale->length), new_al_len - ((u8*) ale - base_ni->attr_list));
|
||||||
|
|
||||||
/* Set new runlist. */
|
/* Set new runlist. */
|
||||||
free(base_ni->attr_list);
|
free(base_ni->attr_list);
|
||||||
base_ni->attr_list = new_al;
|
base_ni->attr_list = new_al;
|
||||||
base_ni->attr_list_size = new_al_len;
|
base_ni->attr_list_size = new_al_len;
|
||||||
NInoAttrListSetDirty(base_ni);
|
NInoAttrListSetDirty(base_ni);
|
||||||
/* Done! */
|
/* Done! */
|
||||||
ntfs_attr_close(na);
|
ntfs_attr_close(na);
|
||||||
return 0;
|
return 0;
|
||||||
err_out:
|
err_out: if (na) ntfs_attr_close(na);
|
||||||
if (na)
|
free(new_al);
|
||||||
ntfs_attr_close(na);
|
errno = err;
|
||||||
free(new_al);
|
return -1;
|
||||||
errno = err;
|
|
||||||
return -1;
|
|
||||||
}
|
}
|
||||||
|
@ -42,10 +42,9 @@ extern int ntfs_attrlist_entry_rm(ntfs_attr_search_ctx *ctx);
|
|||||||
*/
|
*/
|
||||||
static __inline__ void ntfs_attrlist_mark_dirty(ntfs_inode *ni)
|
static __inline__ void ntfs_attrlist_mark_dirty(ntfs_inode *ni)
|
||||||
{
|
{
|
||||||
if (ni->nr_extents == -1)
|
if (ni->nr_extents == -1)
|
||||||
NInoAttrListSetDirty(ni->base_ni);
|
NInoAttrListSetDirty(ni->base_ni);
|
||||||
else
|
else NInoAttrListSetDirty(ni);
|
||||||
NInoAttrListSetDirty(ni);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif /* defined _NTFS_ATTRLIST_H */
|
#endif /* defined _NTFS_ATTRLIST_H */
|
||||||
|
@ -3,17 +3,17 @@
|
|||||||
Functions for dealing with conversion of data between types
|
Functions for dealing with conversion of data between types
|
||||||
|
|
||||||
Copyright (c) 2006 Michael "Chishm" Chisholm
|
Copyright (c) 2006 Michael "Chishm" Chisholm
|
||||||
|
|
||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -24,7 +24,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef _BIT_OPS_H
|
#ifndef _BIT_OPS_H
|
||||||
#define _BIT_OPS_H
|
#define _BIT_OPS_H
|
||||||
@ -32,26 +32,30 @@
|
|||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
/*-----------------------------------------------------------------
|
/*-----------------------------------------------------------------
|
||||||
Functions to deal with little endian values stored in uint8_t arrays
|
Functions to deal with little endian values stored in uint8_t arrays
|
||||||
-----------------------------------------------------------------*/
|
-----------------------------------------------------------------*/
|
||||||
static inline uint16_t u8array_to_u16 (const uint8_t* item, int offset) {
|
static inline uint16_t u8array_to_u16(const uint8_t* item, int offset)
|
||||||
return ( item[offset] | (item[offset + 1] << 8));
|
{
|
||||||
|
return (item[offset] | (item[offset + 1] << 8));
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline uint32_t u8array_to_u32 (const uint8_t* item, int offset) {
|
static inline uint32_t u8array_to_u32(const uint8_t* item, int offset)
|
||||||
return ( item[offset] | (item[offset + 1] << 8) | (item[offset + 2] << 16) | (item[offset + 3] << 24));
|
{
|
||||||
|
return (item[offset] | (item[offset + 1] << 8) | (item[offset + 2] << 16) | (item[offset + 3] << 24));
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void u16_to_u8array (uint8_t* item, int offset, uint16_t value) {
|
static inline void u16_to_u8array(uint8_t* item, int offset, uint16_t value)
|
||||||
item[offset] = (uint8_t) value;
|
{
|
||||||
item[offset + 1] = (uint8_t)(value >> 8);
|
item[offset] = (uint8_t) value;
|
||||||
|
item[offset + 1] = (uint8_t) (value >> 8);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void u32_to_u8array (uint8_t* item, int offset, uint32_t value) {
|
static inline void u32_to_u8array(uint8_t* item, int offset, uint32_t value)
|
||||||
item[offset] = (uint8_t) value;
|
{
|
||||||
item[offset + 1] = (uint8_t)(value >> 8);
|
item[offset] = (uint8_t) value;
|
||||||
item[offset + 2] = (uint8_t)(value >> 16);
|
item[offset + 1] = (uint8_t) (value >> 8);
|
||||||
item[offset + 3] = (uint8_t)(value >> 24);
|
item[offset + 2] = (uint8_t) (value >> 16);
|
||||||
|
item[offset + 3] = (uint8_t) (value >> 24);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif // _BIT_OPS_H
|
#endif // _BIT_OPS_H
|
||||||
|
@ -55,12 +55,10 @@
|
|||||||
*/
|
*/
|
||||||
void ntfs_bit_set(u8 *bitmap, const u64 bit, const u8 new_value)
|
void ntfs_bit_set(u8 *bitmap, const u64 bit, const u8 new_value)
|
||||||
{
|
{
|
||||||
if (!bitmap || new_value > 1)
|
if (!bitmap || new_value > 1) return;
|
||||||
return;
|
if (!new_value)
|
||||||
if (!new_value)
|
bitmap[bit >> 3] &= ~(1 << (bit & 7));
|
||||||
bitmap[bit >> 3] &= ~(1 << (bit & 7));
|
else bitmap[bit >> 3] |= (1 << (bit & 7));
|
||||||
else
|
|
||||||
bitmap[bit >> 3] |= (1 << (bit & 7));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -73,9 +71,8 @@ void ntfs_bit_set(u8 *bitmap, const u64 bit, const u8 new_value)
|
|||||||
*/
|
*/
|
||||||
char ntfs_bit_get(const u8 *bitmap, const u64 bit)
|
char ntfs_bit_get(const u8 *bitmap, const u64 bit)
|
||||||
{
|
{
|
||||||
if (!bitmap)
|
if (!bitmap) return -1;
|
||||||
return -1;
|
return (bitmap[bit >> 3] >> (bit & 7)) & 1;
|
||||||
return (bitmap[bit >> 3] >> (bit & 7)) & 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -89,15 +86,13 @@ char ntfs_bit_get(const u8 *bitmap, const u64 bit)
|
|||||||
*/
|
*/
|
||||||
char ntfs_bit_get_and_set(u8 *bitmap, const u64 bit, const u8 new_value)
|
char ntfs_bit_get_and_set(u8 *bitmap, const u64 bit, const u8 new_value)
|
||||||
{
|
{
|
||||||
register u8 old_bit, shift;
|
register u8 old_bit, shift;
|
||||||
|
|
||||||
if (!bitmap || new_value > 1)
|
if (!bitmap || new_value > 1) return -1;
|
||||||
return -1;
|
shift = bit & 7;
|
||||||
shift = bit & 7;
|
old_bit = (bitmap[bit >> 3] >> shift) & 1;
|
||||||
old_bit = (bitmap[bit >> 3] >> shift) & 1;
|
if (new_value != old_bit) bitmap[bit >> 3] ^= 1 << shift;
|
||||||
if (new_value != old_bit)
|
return old_bit;
|
||||||
bitmap[bit >> 3] ^= 1 << shift;
|
|
||||||
return old_bit;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -112,146 +107,147 @@ char ntfs_bit_get_and_set(u8 *bitmap, const u64 bit, const u8 new_value)
|
|||||||
*
|
*
|
||||||
* On success return 0 and on error return -1 with errno set to the error code.
|
* On success return 0 and on error return -1 with errno set to the error code.
|
||||||
*/
|
*/
|
||||||
static int ntfs_bitmap_set_bits_in_run(ntfs_attr *na, s64 start_bit,
|
static int ntfs_bitmap_set_bits_in_run(ntfs_attr *na, s64 start_bit, s64 count, int value)
|
||||||
s64 count, int value)
|
|
||||||
{
|
{
|
||||||
s64 bufsize, br;
|
s64 bufsize, br;
|
||||||
u8 *buf, *lastbyte_buf;
|
u8 *buf, *lastbyte_buf;
|
||||||
int bit, firstbyte, lastbyte, lastbyte_pos, tmp, ret = -1;
|
int bit, firstbyte, lastbyte, lastbyte_pos, tmp, ret = -1;
|
||||||
|
|
||||||
if (!na || start_bit < 0 || count < 0) {
|
if (!na || start_bit < 0 || count < 0)
|
||||||
errno = EINVAL;
|
{
|
||||||
ntfs_log_perror("%s: Invalid argument (%p, %lld, %lld)",
|
errno = EINVAL;
|
||||||
__FUNCTION__, na, (long long)start_bit, (long long)count);
|
ntfs_log_perror("%s: Invalid argument (%p, %lld, %lld)",
|
||||||
return -1;
|
__FUNCTION__, na, (long long)start_bit, (long long)count);
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
bit = start_bit & 7;
|
bit = start_bit & 7;
|
||||||
if (bit)
|
if (bit)
|
||||||
firstbyte = 1;
|
firstbyte = 1;
|
||||||
else
|
else firstbyte = 0;
|
||||||
firstbyte = 0;
|
|
||||||
|
|
||||||
/* Calculate the required buffer size in bytes, capping it at 8kiB. */
|
/* Calculate the required buffer size in bytes, capping it at 8kiB. */
|
||||||
bufsize = ((count - (bit ? 8 - bit : 0) + 7) >> 3) + firstbyte;
|
bufsize = ((count - (bit ? 8 - bit : 0) + 7) >> 3) + firstbyte;
|
||||||
if (bufsize > 8192)
|
if (bufsize > 8192) bufsize = 8192;
|
||||||
bufsize = 8192;
|
|
||||||
|
|
||||||
buf = ntfs_malloc(bufsize);
|
buf = ntfs_malloc(bufsize);
|
||||||
if (!buf)
|
if (!buf) return -1;
|
||||||
return -1;
|
|
||||||
|
|
||||||
/* Depending on @value, zero or set all bits in the allocated buffer. */
|
|
||||||
memset(buf, value ? 0xff : 0, bufsize);
|
|
||||||
|
|
||||||
/* If there is a first partial byte... */
|
/* Depending on @value, zero or set all bits in the allocated buffer. */
|
||||||
if (bit) {
|
memset(buf, value ? 0xff : 0, bufsize);
|
||||||
/* read it in... */
|
|
||||||
br = ntfs_attr_pread(na, start_bit >> 3, 1, buf);
|
|
||||||
if (br != 1) {
|
|
||||||
if (br >= 0)
|
|
||||||
errno = EIO;
|
|
||||||
goto free_err_out;
|
|
||||||
}
|
|
||||||
/* and set or clear the appropriate bits in it. */
|
|
||||||
while ((bit & 7) && count--) {
|
|
||||||
if (value)
|
|
||||||
*buf |= 1 << bit++;
|
|
||||||
else
|
|
||||||
*buf &= ~(1 << bit++);
|
|
||||||
}
|
|
||||||
/* Update @start_bit to the new position. */
|
|
||||||
start_bit = (start_bit + 7) & ~7;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Loop until @count reaches zero. */
|
/* If there is a first partial byte... */
|
||||||
lastbyte = 0;
|
if (bit)
|
||||||
lastbyte_buf = NULL;
|
{
|
||||||
bit = count & 7;
|
/* read it in... */
|
||||||
do {
|
br = ntfs_attr_pread(na, start_bit >> 3, 1, buf);
|
||||||
/* If there is a last partial byte... */
|
if (br != 1)
|
||||||
if (count > 0 && bit) {
|
{
|
||||||
lastbyte_pos = ((count + 7) >> 3) + firstbyte;
|
if (br >= 0) errno = EIO;
|
||||||
if (!lastbyte_pos) {
|
goto free_err_out;
|
||||||
// FIXME: Eeek! BUG!
|
}
|
||||||
ntfs_log_error("Lastbyte is zero. Leaving "
|
/* and set or clear the appropriate bits in it. */
|
||||||
"inconsistent metadata.\n");
|
while ((bit & 7) && count--)
|
||||||
errno = EIO;
|
{
|
||||||
goto free_err_out;
|
if (value)
|
||||||
}
|
*buf |= 1 << bit++;
|
||||||
/* and it is in the currently loaded bitmap window... */
|
else *buf &= ~(1 << bit++);
|
||||||
if (lastbyte_pos <= bufsize) {
|
}
|
||||||
lastbyte_buf = buf + lastbyte_pos - 1;
|
/* Update @start_bit to the new position. */
|
||||||
|
start_bit = (start_bit + 7) & ~7;
|
||||||
|
}
|
||||||
|
|
||||||
/* read the byte in... */
|
/* Loop until @count reaches zero. */
|
||||||
br = ntfs_attr_pread(na, (start_bit + count) >>
|
lastbyte = 0;
|
||||||
3, 1, lastbyte_buf);
|
lastbyte_buf = NULL;
|
||||||
if (br != 1) {
|
bit = count & 7;
|
||||||
// FIXME: Eeek! We need rollback! (AIA)
|
do
|
||||||
if (br >= 0)
|
{
|
||||||
errno = EIO;
|
/* If there is a last partial byte... */
|
||||||
ntfs_log_perror("Reading of last byte "
|
if (count > 0 && bit)
|
||||||
"failed (%lld). Leaving inconsistent "
|
{
|
||||||
"metadata", (long long)br);
|
lastbyte_pos = ((count + 7) >> 3) + firstbyte;
|
||||||
goto free_err_out;
|
if (!lastbyte_pos)
|
||||||
}
|
{
|
||||||
/* and set/clear the appropriate bits in it. */
|
// FIXME: Eeek! BUG!
|
||||||
while (bit && count--) {
|
ntfs_log_error("Lastbyte is zero. Leaving "
|
||||||
if (value)
|
"inconsistent metadata.\n");
|
||||||
*lastbyte_buf |= 1 << --bit;
|
errno = EIO;
|
||||||
else
|
goto free_err_out;
|
||||||
*lastbyte_buf &= ~(1 << --bit);
|
}
|
||||||
}
|
/* and it is in the currently loaded bitmap window... */
|
||||||
/* We don't want to come back here... */
|
if (lastbyte_pos <= bufsize)
|
||||||
bit = 0;
|
{
|
||||||
/* We have a last byte that we have handled. */
|
lastbyte_buf = buf + lastbyte_pos - 1;
|
||||||
lastbyte = 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Write the prepared buffer to disk. */
|
/* read the byte in... */
|
||||||
tmp = (start_bit >> 3) - firstbyte;
|
br = ntfs_attr_pread(na, (start_bit + count) >> 3, 1, lastbyte_buf);
|
||||||
br = ntfs_attr_pwrite(na, tmp, bufsize, buf);
|
if (br != 1)
|
||||||
if (br != bufsize) {
|
{
|
||||||
// FIXME: Eeek! We need rollback! (AIA)
|
// FIXME: Eeek! We need rollback! (AIA)
|
||||||
if (br >= 0)
|
if (br >= 0) errno = EIO;
|
||||||
errno = EIO;
|
ntfs_log_perror("Reading of last byte "
|
||||||
ntfs_log_perror("Failed to write buffer to bitmap "
|
"failed (%lld). Leaving inconsistent "
|
||||||
"(%lld != %lld). Leaving inconsistent metadata",
|
"metadata", (long long)br);
|
||||||
(long long)br, (long long)bufsize);
|
goto free_err_out;
|
||||||
goto free_err_out;
|
}
|
||||||
}
|
/* and set/clear the appropriate bits in it. */
|
||||||
|
while (bit && count--)
|
||||||
|
{
|
||||||
|
if (value)
|
||||||
|
*lastbyte_buf |= 1 << --bit;
|
||||||
|
else *lastbyte_buf &= ~(1 << --bit);
|
||||||
|
}
|
||||||
|
/* We don't want to come back here... */
|
||||||
|
bit = 0;
|
||||||
|
/* We have a last byte that we have handled. */
|
||||||
|
lastbyte = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Update counters. */
|
/* Write the prepared buffer to disk. */
|
||||||
tmp = (bufsize - firstbyte - lastbyte) << 3;
|
tmp = (start_bit >> 3) - firstbyte;
|
||||||
if (firstbyte) {
|
br = ntfs_attr_pwrite(na, tmp, bufsize, buf);
|
||||||
firstbyte = 0;
|
if (br != bufsize)
|
||||||
/*
|
{
|
||||||
* Re-set the partial first byte so a subsequent write
|
// FIXME: Eeek! We need rollback! (AIA)
|
||||||
* of the buffer does not have stale, incorrect bits.
|
if (br >= 0) errno = EIO;
|
||||||
*/
|
ntfs_log_perror("Failed to write buffer to bitmap "
|
||||||
*buf = value ? 0xff : 0;
|
"(%lld != %lld). Leaving inconsistent metadata",
|
||||||
}
|
(long long)br, (long long)bufsize);
|
||||||
start_bit += tmp;
|
goto free_err_out;
|
||||||
count -= tmp;
|
}
|
||||||
if (bufsize > (tmp = (count + 7) >> 3))
|
|
||||||
bufsize = tmp;
|
|
||||||
|
|
||||||
if (lastbyte && count != 0) {
|
/* Update counters. */
|
||||||
// FIXME: Eeek! BUG!
|
tmp = (bufsize - firstbyte - lastbyte) << 3;
|
||||||
ntfs_log_error("Last buffer but count is not zero "
|
if (firstbyte)
|
||||||
"(%lld). Leaving inconsistent metadata.\n",
|
{
|
||||||
(long long)count);
|
firstbyte = 0;
|
||||||
errno = EIO;
|
/*
|
||||||
goto free_err_out;
|
* Re-set the partial first byte so a subsequent write
|
||||||
}
|
* of the buffer does not have stale, incorrect bits.
|
||||||
} while (count > 0);
|
*/
|
||||||
|
*buf = value ? 0xff : 0;
|
||||||
ret = 0;
|
}
|
||||||
|
start_bit += tmp;
|
||||||
free_err_out:
|
count -= tmp;
|
||||||
free(buf);
|
if (bufsize > (tmp = (count + 7) >> 3)) bufsize = tmp;
|
||||||
return ret;
|
|
||||||
|
if (lastbyte && count != 0)
|
||||||
|
{
|
||||||
|
// FIXME: Eeek! BUG!
|
||||||
|
ntfs_log_error("Last buffer but count is not zero "
|
||||||
|
"(%lld). Leaving inconsistent metadata.\n",
|
||||||
|
(long long)count);
|
||||||
|
errno = EIO;
|
||||||
|
goto free_err_out;
|
||||||
|
}
|
||||||
|
} while (count > 0);
|
||||||
|
|
||||||
|
ret = 0;
|
||||||
|
|
||||||
|
free_err_out: free(buf);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -267,13 +263,13 @@ free_err_out:
|
|||||||
*/
|
*/
|
||||||
int ntfs_bitmap_set_run(ntfs_attr *na, s64 start_bit, s64 count)
|
int ntfs_bitmap_set_run(ntfs_attr *na, s64 start_bit, s64 count)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ntfs_log_enter("Set from bit %lld, count %lld\n",
|
ntfs_log_enter("Set from bit %lld, count %lld\n",
|
||||||
(long long)start_bit, (long long)count);
|
(long long)start_bit, (long long)count);
|
||||||
ret = ntfs_bitmap_set_bits_in_run(na, start_bit, count, 1);
|
ret = ntfs_bitmap_set_bits_in_run(na, start_bit, count, 1);
|
||||||
ntfs_log_leave("\n");
|
ntfs_log_leave("\n");
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -289,12 +285,12 @@ int ntfs_bitmap_set_run(ntfs_attr *na, s64 start_bit, s64 count)
|
|||||||
*/
|
*/
|
||||||
int ntfs_bitmap_clear_run(ntfs_attr *na, s64 start_bit, s64 count)
|
int ntfs_bitmap_clear_run(ntfs_attr *na, s64 start_bit, s64 count)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ntfs_log_enter("Clear from bit %lld, count %lld\n",
|
ntfs_log_enter("Clear from bit %lld, count %lld\n",
|
||||||
(long long)start_bit, (long long)count);
|
(long long)start_bit, (long long)count);
|
||||||
ret = ntfs_bitmap_set_bits_in_run(na, start_bit, count, 0);
|
ret = ntfs_bitmap_set_bits_in_run(na, start_bit, count, 0);
|
||||||
ntfs_log_leave("\n");
|
ntfs_log_leave("\n");
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,8 +39,8 @@
|
|||||||
extern void ntfs_bit_set(u8 *bitmap, const u64 bit, const u8 new_value);
|
extern void ntfs_bit_set(u8 *bitmap, const u64 bit, const u8 new_value);
|
||||||
extern char ntfs_bit_get(const u8 *bitmap, const u64 bit);
|
extern char ntfs_bit_get(const u8 *bitmap, const u64 bit);
|
||||||
extern char ntfs_bit_get_and_set(u8 *bitmap, const u64 bit, const u8 new_value);
|
extern char ntfs_bit_get_and_set(u8 *bitmap, const u64 bit, const u8 new_value);
|
||||||
extern int ntfs_bitmap_set_run(ntfs_attr *na, s64 start_bit, s64 count);
|
extern int ntfs_bitmap_set_run(ntfs_attr *na, s64 start_bit, s64 count);
|
||||||
extern int ntfs_bitmap_clear_run(ntfs_attr *na, s64 start_bit, s64 count);
|
extern int ntfs_bitmap_clear_run(ntfs_attr *na, s64 start_bit, s64 count);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ntfs_bitmap_set_bit - set a bit in a bitmap
|
* ntfs_bitmap_set_bit - set a bit in a bitmap
|
||||||
@ -53,7 +53,7 @@ extern int ntfs_bitmap_clear_run(ntfs_attr *na, s64 start_bit, s64 count);
|
|||||||
*/
|
*/
|
||||||
static __inline__ int ntfs_bitmap_set_bit(ntfs_attr *na, s64 bit)
|
static __inline__ int ntfs_bitmap_set_bit(ntfs_attr *na, s64 bit)
|
||||||
{
|
{
|
||||||
return ntfs_bitmap_set_run(na, bit, 1);
|
return ntfs_bitmap_set_run(na, bit, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -67,7 +67,7 @@ static __inline__ int ntfs_bitmap_set_bit(ntfs_attr *na, s64 bit)
|
|||||||
*/
|
*/
|
||||||
static __inline__ int ntfs_bitmap_clear_bit(ntfs_attr *na, s64 bit)
|
static __inline__ int ntfs_bitmap_clear_bit(ntfs_attr *na, s64 bit)
|
||||||
{
|
{
|
||||||
return ntfs_bitmap_clear_run(na, bit, 1);
|
return ntfs_bitmap_clear_run(na, bit, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -78,7 +78,7 @@ static __inline__ int ntfs_bitmap_clear_bit(ntfs_attr *na, s64 bit)
|
|||||||
*/
|
*/
|
||||||
static __inline__ u32 ntfs_rol32(u32 word, unsigned int shift)
|
static __inline__ u32 ntfs_rol32(u32 word, unsigned int shift)
|
||||||
{
|
{
|
||||||
return (word << shift) | (word >> (32 - shift));
|
return (word << shift) | (word >> (32 - shift));
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -89,7 +89,7 @@ static __inline__ u32 ntfs_rol32(u32 word, unsigned int shift)
|
|||||||
*/
|
*/
|
||||||
static __inline__ u32 ntfs_ror32(u32 word, unsigned int shift)
|
static __inline__ u32 ntfs_ror32(u32 word, unsigned int shift)
|
||||||
{
|
{
|
||||||
return (word >> shift) | (word << (32 - shift));
|
return (word >> shift) | (word << (32 - shift));
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif /* defined _NTFS_BITMAP_H */
|
#endif /* defined _NTFS_BITMAP_H */
|
||||||
|
@ -59,104 +59,122 @@
|
|||||||
*/
|
*/
|
||||||
BOOL ntfs_boot_sector_is_ntfs(NTFS_BOOT_SECTOR *b)
|
BOOL ntfs_boot_sector_is_ntfs(NTFS_BOOT_SECTOR *b)
|
||||||
{
|
{
|
||||||
u32 i;
|
u32 i;
|
||||||
BOOL ret = FALSE;
|
BOOL ret = FALSE;
|
||||||
|
|
||||||
ntfs_log_debug("Beginning bootsector check.\n");
|
ntfs_log_debug("Beginning bootsector check.\n");
|
||||||
|
|
||||||
ntfs_log_debug("Checking OEMid, NTFS signature.\n");
|
ntfs_log_debug("Checking OEMid, NTFS signature.\n");
|
||||||
if (b->oem_id != cpu_to_le64(0x202020205346544eULL)) { /* "NTFS " */
|
if (b->oem_id != cpu_to_le64(0x202020205346544eULL))
|
||||||
ntfs_log_error("NTFS signature is missing.\n");
|
{ /* "NTFS " */
|
||||||
goto not_ntfs;
|
ntfs_log_error("NTFS signature is missing.\n");
|
||||||
}
|
goto not_ntfs;
|
||||||
|
}
|
||||||
|
|
||||||
ntfs_log_debug("Checking bytes per sector.\n");
|
ntfs_log_debug("Checking bytes per sector.\n");
|
||||||
if (le16_to_cpu(b->bpb.bytes_per_sector) < 256 ||
|
if (le16_to_cpu(b->bpb.bytes_per_sector) < 256 || le16_to_cpu(b->bpb.bytes_per_sector) > 4096)
|
||||||
le16_to_cpu(b->bpb.bytes_per_sector) > 4096) {
|
{
|
||||||
ntfs_log_error("Unexpected bytes per sector value (%d).\n",
|
ntfs_log_error("Unexpected bytes per sector value (%d).\n",
|
||||||
le16_to_cpu(b->bpb.bytes_per_sector));
|
le16_to_cpu(b->bpb.bytes_per_sector));
|
||||||
goto not_ntfs;
|
goto not_ntfs;
|
||||||
}
|
}
|
||||||
|
|
||||||
ntfs_log_debug("Checking sectors per cluster.\n");
|
ntfs_log_debug("Checking sectors per cluster.\n");
|
||||||
switch (b->bpb.sectors_per_cluster) {
|
switch (b->bpb.sectors_per_cluster)
|
||||||
case 1: case 2: case 4: case 8: case 16: case 32: case 64: case 128:
|
{
|
||||||
break;
|
case 1:
|
||||||
default:
|
case 2:
|
||||||
ntfs_log_error("Unexpected sectors per cluster value (%d).\n",
|
case 4:
|
||||||
b->bpb.sectors_per_cluster);
|
case 8:
|
||||||
goto not_ntfs;
|
case 16:
|
||||||
}
|
case 32:
|
||||||
|
case 64:
|
||||||
|
case 128:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ntfs_log_error("Unexpected sectors per cluster value (%d).\n",
|
||||||
|
b->bpb.sectors_per_cluster);
|
||||||
|
goto not_ntfs;
|
||||||
|
}
|
||||||
|
|
||||||
ntfs_log_debug("Checking cluster size.\n");
|
ntfs_log_debug("Checking cluster size.\n");
|
||||||
i = (u32)le16_to_cpu(b->bpb.bytes_per_sector) *
|
i = (u32) le16_to_cpu(b->bpb.bytes_per_sector) * b->bpb.sectors_per_cluster;
|
||||||
b->bpb.sectors_per_cluster;
|
if (i > 65536)
|
||||||
if (i > 65536) {
|
{
|
||||||
ntfs_log_error("Unexpected cluster size (%d).\n", i);
|
ntfs_log_error("Unexpected cluster size (%d).\n", i);
|
||||||
goto not_ntfs;
|
goto not_ntfs;
|
||||||
}
|
}
|
||||||
|
|
||||||
ntfs_log_debug("Checking reserved fields are zero.\n");
|
ntfs_log_debug("Checking reserved fields are zero.\n");
|
||||||
if (le16_to_cpu(b->bpb.reserved_sectors) ||
|
if (le16_to_cpu(b->bpb.reserved_sectors) || le16_to_cpu(b->bpb.root_entries) || le16_to_cpu(b->bpb.sectors)
|
||||||
le16_to_cpu(b->bpb.root_entries) ||
|
|| le16_to_cpu(b->bpb.sectors_per_fat) || le32_to_cpu(b->bpb.large_sectors) || b->bpb.fats)
|
||||||
le16_to_cpu(b->bpb.sectors) ||
|
{
|
||||||
le16_to_cpu(b->bpb.sectors_per_fat) ||
|
ntfs_log_error("Reserved fields aren't zero "
|
||||||
le32_to_cpu(b->bpb.large_sectors) ||
|
"(%d, %d, %d, %d, %d, %d).\n",
|
||||||
b->bpb.fats) {
|
le16_to_cpu(b->bpb.reserved_sectors),
|
||||||
ntfs_log_error("Reserved fields aren't zero "
|
le16_to_cpu(b->bpb.root_entries),
|
||||||
"(%d, %d, %d, %d, %d, %d).\n",
|
le16_to_cpu(b->bpb.sectors),
|
||||||
le16_to_cpu(b->bpb.reserved_sectors),
|
le16_to_cpu(b->bpb.sectors_per_fat),
|
||||||
le16_to_cpu(b->bpb.root_entries),
|
le32_to_cpu(b->bpb.large_sectors),
|
||||||
le16_to_cpu(b->bpb.sectors),
|
b->bpb.fats);
|
||||||
le16_to_cpu(b->bpb.sectors_per_fat),
|
goto not_ntfs;
|
||||||
le32_to_cpu(b->bpb.large_sectors),
|
}
|
||||||
b->bpb.fats);
|
|
||||||
goto not_ntfs;
|
|
||||||
}
|
|
||||||
|
|
||||||
ntfs_log_debug("Checking clusters per mft record.\n");
|
ntfs_log_debug("Checking clusters per mft record.\n");
|
||||||
if ((u8)b->clusters_per_mft_record < 0xe1 ||
|
if ((u8) b->clusters_per_mft_record < 0xe1 || (u8) b->clusters_per_mft_record > 0xf7)
|
||||||
(u8)b->clusters_per_mft_record > 0xf7) {
|
{
|
||||||
switch (b->clusters_per_mft_record) {
|
switch (b->clusters_per_mft_record)
|
||||||
case 1: case 2: case 4: case 8: case 0x10: case 0x20: case 0x40:
|
{
|
||||||
break;
|
case 1:
|
||||||
default:
|
case 2:
|
||||||
ntfs_log_error("Unexpected clusters per mft record "
|
case 4:
|
||||||
"(%d).\n", b->clusters_per_mft_record);
|
case 8:
|
||||||
goto not_ntfs;
|
case 0x10:
|
||||||
}
|
case 0x20:
|
||||||
}
|
case 0x40:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ntfs_log_error("Unexpected clusters per mft record "
|
||||||
|
"(%d).\n", b->clusters_per_mft_record);
|
||||||
|
goto not_ntfs;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ntfs_log_debug("Checking clusters per index block.\n");
|
ntfs_log_debug("Checking clusters per index block.\n");
|
||||||
if ((u8)b->clusters_per_index_record < 0xe1 ||
|
if ((u8) b->clusters_per_index_record < 0xe1 || (u8) b->clusters_per_index_record > 0xf7)
|
||||||
(u8)b->clusters_per_index_record > 0xf7) {
|
{
|
||||||
switch (b->clusters_per_index_record) {
|
switch (b->clusters_per_index_record)
|
||||||
case 1: case 2: case 4: case 8: case 0x10: case 0x20: case 0x40:
|
{
|
||||||
break;
|
case 1:
|
||||||
default:
|
case 2:
|
||||||
ntfs_log_error("Unexpected clusters per index record "
|
case 4:
|
||||||
"(%d).\n", b->clusters_per_index_record);
|
case 8:
|
||||||
goto not_ntfs;
|
case 0x10:
|
||||||
}
|
case 0x20:
|
||||||
}
|
case 0x40:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ntfs_log_error("Unexpected clusters per index record "
|
||||||
|
"(%d).\n", b->clusters_per_index_record);
|
||||||
|
goto not_ntfs;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (b->end_of_sector_marker != cpu_to_le16(0xaa55))
|
if (b->end_of_sector_marker != cpu_to_le16(0xaa55))
|
||||||
ntfs_log_debug("Warning: Bootsector has invalid end of sector "
|
ntfs_log_debug("Warning: Bootsector has invalid end of sector "
|
||||||
"marker.\n");
|
"marker.\n");
|
||||||
|
|
||||||
ntfs_log_debug("Bootsector check completed successfully.\n");
|
ntfs_log_debug("Bootsector check completed successfully.\n");
|
||||||
|
|
||||||
ret = TRUE;
|
ret = TRUE;
|
||||||
not_ntfs:
|
not_ntfs: return ret;
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static const char *last_sector_error =
|
static const char *last_sector_error = "HINTS: Either the volume is a RAID/LDM but it wasn't setup yet,\n"
|
||||||
"HINTS: Either the volume is a RAID/LDM but it wasn't setup yet,\n"
|
" or it was not setup correctly (e.g. by not using mdadm --build ...),\n"
|
||||||
" or it was not setup correctly (e.g. by not using mdadm --build ...),\n"
|
" or a wrong device is tried to be mounted,\n"
|
||||||
" or a wrong device is tried to be mounted,\n"
|
" or the partition table is corrupt (partition is smaller than NTFS),\n"
|
||||||
" or the partition table is corrupt (partition is smaller than NTFS),\n"
|
" or the NTFS boot sector is corrupt (NTFS size is not valid).\n";
|
||||||
" or the NTFS boot sector is corrupt (NTFS size is not valid).\n";
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ntfs_boot_sector_parse - setup an ntfs volume from an ntfs boot sector
|
* ntfs_boot_sector_parse - setup an ntfs volume from an ntfs boot sector
|
||||||
@ -170,116 +188,116 @@ static const char *last_sector_error =
|
|||||||
*/
|
*/
|
||||||
int ntfs_boot_sector_parse(ntfs_volume *vol, const NTFS_BOOT_SECTOR *bs)
|
int ntfs_boot_sector_parse(ntfs_volume *vol, const NTFS_BOOT_SECTOR *bs)
|
||||||
{
|
{
|
||||||
s64 sectors;
|
s64 sectors;
|
||||||
u8 sectors_per_cluster;
|
u8 sectors_per_cluster;
|
||||||
s8 c;
|
s8 c;
|
||||||
|
|
||||||
/* We return -1 with errno = EINVAL on error. */
|
/* We return -1 with errno = EINVAL on error. */
|
||||||
errno = EINVAL;
|
errno = EINVAL;
|
||||||
|
|
||||||
vol->sector_size = le16_to_cpu(bs->bpb.bytes_per_sector);
|
vol->sector_size = le16_to_cpu(bs->bpb.bytes_per_sector);
|
||||||
vol->sector_size_bits = ffs(vol->sector_size) - 1;
|
vol->sector_size_bits = ffs(vol->sector_size) - 1;
|
||||||
ntfs_log_debug("SectorSize = 0x%x\n", vol->sector_size);
|
ntfs_log_debug("SectorSize = 0x%x\n", vol->sector_size);
|
||||||
ntfs_log_debug("SectorSizeBits = %u\n", vol->sector_size_bits);
|
ntfs_log_debug("SectorSizeBits = %u\n", vol->sector_size_bits);
|
||||||
/*
|
/*
|
||||||
* The bounds checks on mft_lcn and mft_mirr_lcn (i.e. them being
|
* The bounds checks on mft_lcn and mft_mirr_lcn (i.e. them being
|
||||||
* below or equal the number_of_clusters) really belong in the
|
* below or equal the number_of_clusters) really belong in the
|
||||||
* ntfs_boot_sector_is_ntfs but in this way we can just do this once.
|
* ntfs_boot_sector_is_ntfs but in this way we can just do this once.
|
||||||
*/
|
*/
|
||||||
sectors_per_cluster = bs->bpb.sectors_per_cluster;
|
sectors_per_cluster = bs->bpb.sectors_per_cluster;
|
||||||
ntfs_log_debug("SectorsPerCluster = 0x%x\n", sectors_per_cluster);
|
ntfs_log_debug("SectorsPerCluster = 0x%x\n", sectors_per_cluster);
|
||||||
if (sectors_per_cluster & (sectors_per_cluster - 1)) {
|
if (sectors_per_cluster & (sectors_per_cluster - 1))
|
||||||
ntfs_log_error("sectors_per_cluster (%d) is not a power of 2."
|
{
|
||||||
"\n", sectors_per_cluster);
|
ntfs_log_error("sectors_per_cluster (%d) is not a power of 2."
|
||||||
return -1;
|
"\n", sectors_per_cluster);
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
sectors = sle64_to_cpu(bs->number_of_sectors);
|
|
||||||
ntfs_log_debug("NumberOfSectors = %lld\n", (long long)sectors);
|
|
||||||
if (!sectors) {
|
|
||||||
ntfs_log_error("Volume size is set to zero.\n");
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
if (vol->dev->d_ops->seek(vol->dev,
|
|
||||||
(sectors - 1) << vol->sector_size_bits,
|
|
||||||
SEEK_SET) == -1) {
|
|
||||||
ntfs_log_perror("Failed to read last sector (%lld)",
|
|
||||||
(long long)sectors);
|
|
||||||
ntfs_log_error("%s", last_sector_error);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
vol->nr_clusters = sectors >> (ffs(sectors_per_cluster) - 1);
|
|
||||||
|
|
||||||
vol->mft_lcn = sle64_to_cpu(bs->mft_lcn);
|
sectors = sle64_to_cpu(bs->number_of_sectors);
|
||||||
vol->mftmirr_lcn = sle64_to_cpu(bs->mftmirr_lcn);
|
ntfs_log_debug("NumberOfSectors = %lld\n", (long long)sectors);
|
||||||
ntfs_log_debug("MFT LCN = %lld\n", (long long)vol->mft_lcn);
|
if (!sectors)
|
||||||
ntfs_log_debug("MFTMirr LCN = %lld\n", (long long)vol->mftmirr_lcn);
|
{
|
||||||
if (vol->mft_lcn > vol->nr_clusters ||
|
ntfs_log_error("Volume size is set to zero.\n");
|
||||||
vol->mftmirr_lcn > vol->nr_clusters) {
|
return -1;
|
||||||
ntfs_log_error("$MFT LCN (%lld) or $MFTMirr LCN (%lld) is "
|
}
|
||||||
"greater than the number of clusters (%lld).\n",
|
if (vol->dev->d_ops->seek(vol->dev, (sectors - 1) << vol->sector_size_bits, SEEK_SET) == -1)
|
||||||
(long long)vol->mft_lcn, (long long)vol->mftmirr_lcn,
|
{
|
||||||
(long long)vol->nr_clusters);
|
ntfs_log_perror("Failed to read last sector (%lld)",
|
||||||
return -1;
|
(long long)sectors);
|
||||||
}
|
ntfs_log_error("%s", last_sector_error);
|
||||||
|
return -1;
|
||||||
vol->cluster_size = sectors_per_cluster * vol->sector_size;
|
}
|
||||||
if (vol->cluster_size & (vol->cluster_size - 1)) {
|
|
||||||
ntfs_log_error("cluster_size (%d) is not a power of 2.\n",
|
vol->nr_clusters = sectors >> (ffs(sectors_per_cluster) - 1);
|
||||||
vol->cluster_size);
|
|
||||||
return -1;
|
vol->mft_lcn = sle64_to_cpu(bs->mft_lcn);
|
||||||
}
|
vol->mftmirr_lcn = sle64_to_cpu(bs->mftmirr_lcn);
|
||||||
vol->cluster_size_bits = ffs(vol->cluster_size) - 1;
|
ntfs_log_debug("MFT LCN = %lld\n", (long long)vol->mft_lcn);
|
||||||
/*
|
ntfs_log_debug("MFTMirr LCN = %lld\n", (long long)vol->mftmirr_lcn);
|
||||||
* Need to get the clusters per mft record and handle it if it is
|
if (vol->mft_lcn > vol->nr_clusters || vol->mftmirr_lcn > vol->nr_clusters)
|
||||||
* negative. Then calculate the mft_record_size. A value of 0x80 is
|
{
|
||||||
* illegal, thus signed char is actually ok!
|
ntfs_log_error("$MFT LCN (%lld) or $MFTMirr LCN (%lld) is "
|
||||||
*/
|
"greater than the number of clusters (%lld).\n",
|
||||||
c = bs->clusters_per_mft_record;
|
(long long)vol->mft_lcn, (long long)vol->mftmirr_lcn,
|
||||||
ntfs_log_debug("ClusterSize = 0x%x\n", (unsigned)vol->cluster_size);
|
(long long)vol->nr_clusters);
|
||||||
ntfs_log_debug("ClusterSizeBits = %u\n", vol->cluster_size_bits);
|
return -1;
|
||||||
ntfs_log_debug("ClustersPerMftRecord = 0x%x\n", c);
|
}
|
||||||
/*
|
|
||||||
* When clusters_per_mft_record is negative, it means that it is to
|
vol->cluster_size = sectors_per_cluster * vol->sector_size;
|
||||||
* be taken to be the negative base 2 logarithm of the mft_record_size
|
if (vol->cluster_size & (vol->cluster_size - 1))
|
||||||
* min bytes. Then:
|
{
|
||||||
* mft_record_size = 2^(-clusters_per_mft_record) bytes.
|
ntfs_log_error("cluster_size (%d) is not a power of 2.\n",
|
||||||
*/
|
vol->cluster_size);
|
||||||
if (c < 0)
|
return -1;
|
||||||
vol->mft_record_size = 1 << -c;
|
}
|
||||||
else
|
vol->cluster_size_bits = ffs(vol->cluster_size) - 1;
|
||||||
vol->mft_record_size = c << vol->cluster_size_bits;
|
/*
|
||||||
if (vol->mft_record_size & (vol->mft_record_size - 1)) {
|
* Need to get the clusters per mft record and handle it if it is
|
||||||
ntfs_log_error("mft_record_size (%d) is not a power of 2.\n",
|
* negative. Then calculate the mft_record_size. A value of 0x80 is
|
||||||
vol->mft_record_size);
|
* illegal, thus signed char is actually ok!
|
||||||
return -1;
|
*/
|
||||||
}
|
c = bs->clusters_per_mft_record;
|
||||||
vol->mft_record_size_bits = ffs(vol->mft_record_size) - 1;
|
ntfs_log_debug("ClusterSize = 0x%x\n", (unsigned)vol->cluster_size);
|
||||||
ntfs_log_debug("MftRecordSize = 0x%x\n", (unsigned)vol->mft_record_size);
|
ntfs_log_debug("ClusterSizeBits = %u\n", vol->cluster_size_bits);
|
||||||
ntfs_log_debug("MftRecordSizeBits = %u\n", vol->mft_record_size_bits);
|
ntfs_log_debug("ClustersPerMftRecord = 0x%x\n", c);
|
||||||
/* Same as above for INDX record. */
|
/*
|
||||||
c = bs->clusters_per_index_record;
|
* When clusters_per_mft_record is negative, it means that it is to
|
||||||
ntfs_log_debug("ClustersPerINDXRecord = 0x%x\n", c);
|
* be taken to be the negative base 2 logarithm of the mft_record_size
|
||||||
if (c < 0)
|
* min bytes. Then:
|
||||||
vol->indx_record_size = 1 << -c;
|
* mft_record_size = 2^(-clusters_per_mft_record) bytes.
|
||||||
else
|
*/
|
||||||
vol->indx_record_size = c << vol->cluster_size_bits;
|
if (c < 0)
|
||||||
vol->indx_record_size_bits = ffs(vol->indx_record_size) - 1;
|
vol->mft_record_size = 1 << -c;
|
||||||
ntfs_log_debug("INDXRecordSize = 0x%x\n", (unsigned)vol->indx_record_size);
|
else vol->mft_record_size = c << vol->cluster_size_bits;
|
||||||
ntfs_log_debug("INDXRecordSizeBits = %u\n", vol->indx_record_size_bits);
|
if (vol->mft_record_size & (vol->mft_record_size - 1))
|
||||||
/*
|
{
|
||||||
* Work out the size of the MFT mirror in number of mft records. If the
|
ntfs_log_error("mft_record_size (%d) is not a power of 2.\n",
|
||||||
* cluster size is less than or equal to the size taken by four mft
|
vol->mft_record_size);
|
||||||
* records, the mft mirror stores the first four mft records. If the
|
return -1;
|
||||||
* cluster size is bigger than the size taken by four mft records, the
|
}
|
||||||
* mft mirror contains as many mft records as will fit into one
|
vol->mft_record_size_bits = ffs(vol->mft_record_size) - 1;
|
||||||
* cluster.
|
ntfs_log_debug("MftRecordSize = 0x%x\n", (unsigned)vol->mft_record_size);
|
||||||
*/
|
ntfs_log_debug("MftRecordSizeBits = %u\n", vol->mft_record_size_bits);
|
||||||
if (vol->cluster_size <= 4 * vol->mft_record_size)
|
/* Same as above for INDX record. */
|
||||||
vol->mftmirr_size = 4;
|
c = bs->clusters_per_index_record;
|
||||||
else
|
ntfs_log_debug("ClustersPerINDXRecord = 0x%x\n", c);
|
||||||
vol->mftmirr_size = vol->cluster_size / vol->mft_record_size;
|
if (c < 0)
|
||||||
return 0;
|
vol->indx_record_size = 1 << -c;
|
||||||
|
else vol->indx_record_size = c << vol->cluster_size_bits;
|
||||||
|
vol->indx_record_size_bits = ffs(vol->indx_record_size) - 1;
|
||||||
|
ntfs_log_debug("INDXRecordSize = 0x%x\n", (unsigned)vol->indx_record_size);
|
||||||
|
ntfs_log_debug("INDXRecordSizeBits = %u\n", vol->indx_record_size_bits);
|
||||||
|
/*
|
||||||
|
* Work out the size of the MFT mirror in number of mft records. If the
|
||||||
|
* cluster size is less than or equal to the size taken by four mft
|
||||||
|
* records, the mft mirror stores the first four mft records. If the
|
||||||
|
* cluster size is bigger than the size taken by four mft records, the
|
||||||
|
* mft mirror contains as many mft records as will fit into one
|
||||||
|
* cluster.
|
||||||
|
*/
|
||||||
|
if (vol->cluster_size <= 4 * vol->mft_record_size)
|
||||||
|
vol->mftmirr_size = 4;
|
||||||
|
else vol->mftmirr_size = vol->cluster_size / vol->mft_record_size;
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -60,81 +60,92 @@
|
|||||||
* Do not call when a record has been modified (with no key change)
|
* Do not call when a record has been modified (with no key change)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static void inserthashindex(struct CACHE_HEADER *cache,
|
static void inserthashindex(struct CACHE_HEADER *cache, struct CACHED_GENERIC *current)
|
||||||
struct CACHED_GENERIC *current)
|
|
||||||
{
|
{
|
||||||
int h;
|
int h;
|
||||||
struct HASH_ENTRY *link;
|
struct HASH_ENTRY *link;
|
||||||
struct HASH_ENTRY *first;
|
struct HASH_ENTRY *first;
|
||||||
|
|
||||||
if (cache->dohash) {
|
if (cache->dohash)
|
||||||
h = cache->dohash(current);
|
{
|
||||||
if ((h >= 0) && (h < cache->max_hash)) {
|
h = cache->dohash(current);
|
||||||
/* get a free link and insert at top of hash list */
|
if ((h >= 0) && (h < cache->max_hash))
|
||||||
link = cache->free_hash;
|
{
|
||||||
if (link) {
|
/* get a free link and insert at top of hash list */
|
||||||
cache->free_hash = link->next;
|
link = cache->free_hash;
|
||||||
first = cache->first_hash[h];
|
if (link)
|
||||||
if (first)
|
{
|
||||||
link->next = first;
|
cache->free_hash = link->next;
|
||||||
else
|
first = cache->first_hash[h];
|
||||||
link->next = NULL;
|
if (first)
|
||||||
link->entry = current;
|
link->next = first;
|
||||||
cache->first_hash[h] = link;
|
else link->next = NULL;
|
||||||
} else {
|
link->entry = current;
|
||||||
ntfs_log_error("No more hash entries,"
|
cache->first_hash[h] = link;
|
||||||
" cache %s hashing dropped\n",
|
}
|
||||||
cache->name);
|
else
|
||||||
cache->dohash = (cache_hash)NULL;
|
{
|
||||||
}
|
ntfs_log_error("No more hash entries,"
|
||||||
} else {
|
" cache %s hashing dropped\n",
|
||||||
ntfs_log_error("Illegal hash value,"
|
cache->name);
|
||||||
" cache %s hashing dropped\n",
|
cache->dohash = (cache_hash) NULL;
|
||||||
cache->name);
|
}
|
||||||
cache->dohash = (cache_hash)NULL;
|
}
|
||||||
}
|
else
|
||||||
}
|
{
|
||||||
|
ntfs_log_error("Illegal hash value,"
|
||||||
|
" cache %s hashing dropped\n",
|
||||||
|
cache->name);
|
||||||
|
cache->dohash = (cache_hash) NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Drop a hash index when a record is about to be deleted
|
* Drop a hash index when a record is about to be deleted
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static void drophashindex(struct CACHE_HEADER *cache,
|
static void drophashindex(struct CACHE_HEADER *cache, const struct CACHED_GENERIC *current, int hash)
|
||||||
const struct CACHED_GENERIC *current, int hash)
|
|
||||||
{
|
{
|
||||||
struct HASH_ENTRY *link;
|
struct HASH_ENTRY *link;
|
||||||
struct HASH_ENTRY *previous;
|
struct HASH_ENTRY *previous;
|
||||||
|
|
||||||
if (cache->dohash) {
|
if (cache->dohash)
|
||||||
if ((hash >= 0) && (hash < cache->max_hash)) {
|
{
|
||||||
/* find the link and unlink */
|
if ((hash >= 0) && (hash < cache->max_hash))
|
||||||
link = cache->first_hash[hash];
|
{
|
||||||
previous = (struct HASH_ENTRY*)NULL;
|
/* find the link and unlink */
|
||||||
while (link && (link->entry != current)) {
|
link = cache->first_hash[hash];
|
||||||
previous = link;
|
previous = (struct HASH_ENTRY*) NULL;
|
||||||
link = link->next;
|
while (link && (link->entry != current))
|
||||||
}
|
{
|
||||||
if (link) {
|
previous = link;
|
||||||
if (previous)
|
link = link->next;
|
||||||
previous->next = link->next;
|
}
|
||||||
else
|
if (link)
|
||||||
cache->first_hash[hash] = link->next;
|
{
|
||||||
link->next = cache->free_hash;
|
if (previous)
|
||||||
cache->free_hash = link;
|
previous->next = link->next;
|
||||||
} else {
|
else cache->first_hash[hash] = link->next;
|
||||||
ntfs_log_error("Bad hash list,"
|
link->next = cache->free_hash;
|
||||||
" cache %s hashing dropped\n",
|
cache->free_hash = link;
|
||||||
cache->name);
|
}
|
||||||
cache->dohash = (cache_hash)NULL;
|
else
|
||||||
}
|
{
|
||||||
} else {
|
ntfs_log_error("Bad hash list,"
|
||||||
ntfs_log_error("Illegal hash value,"
|
" cache %s hashing dropped\n",
|
||||||
" cache %s hashing dropped\n",
|
cache->name);
|
||||||
cache->name);
|
cache->dohash = (cache_hash) NULL;
|
||||||
cache->dohash = (cache_hash)NULL;
|
}
|
||||||
}
|
}
|
||||||
}
|
else
|
||||||
|
{
|
||||||
|
ntfs_log_error("Illegal hash value,"
|
||||||
|
" cache %s hashing dropped\n",
|
||||||
|
cache->name);
|
||||||
|
cache->dohash = (cache_hash) NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -144,64 +155,64 @@ static void drophashindex(struct CACHE_HEADER *cache,
|
|||||||
* The returned entry may be modified, but not freed
|
* The returned entry may be modified, but not freed
|
||||||
*/
|
*/
|
||||||
|
|
||||||
struct CACHED_GENERIC *ntfs_fetch_cache(struct CACHE_HEADER *cache,
|
struct CACHED_GENERIC *ntfs_fetch_cache(struct CACHE_HEADER *cache, const struct CACHED_GENERIC *wanted,
|
||||||
const struct CACHED_GENERIC *wanted, cache_compare compare)
|
cache_compare compare)
|
||||||
{
|
{
|
||||||
struct CACHED_GENERIC *current;
|
struct CACHED_GENERIC *current;
|
||||||
struct CACHED_GENERIC *previous;
|
struct CACHED_GENERIC *previous;
|
||||||
struct HASH_ENTRY *link;
|
struct HASH_ENTRY *link;
|
||||||
int h;
|
int h;
|
||||||
|
|
||||||
current = (struct CACHED_GENERIC*)NULL;
|
current = (struct CACHED_GENERIC*) NULL;
|
||||||
if (cache) {
|
if (cache)
|
||||||
if (cache->dohash) {
|
{
|
||||||
/*
|
if (cache->dohash)
|
||||||
* When possible, use the hash table to
|
{
|
||||||
* locate the entry if present
|
/*
|
||||||
*/
|
* When possible, use the hash table to
|
||||||
h = cache->dohash(wanted);
|
* locate the entry if present
|
||||||
link = cache->first_hash[h];
|
*/
|
||||||
while (link && compare(link->entry, wanted))
|
h = cache->dohash(wanted);
|
||||||
link = link->next;
|
link = cache->first_hash[h];
|
||||||
if (link)
|
while (link && compare(link->entry, wanted))
|
||||||
current = link->entry;
|
link = link->next;
|
||||||
}
|
if (link) current = link->entry;
|
||||||
if (!cache->dohash) {
|
}
|
||||||
/*
|
if (!cache->dohash)
|
||||||
* Search sequentially in LRU list if no hash table
|
{
|
||||||
* or if hashing has just failed
|
/*
|
||||||
*/
|
* Search sequentially in LRU list if no hash table
|
||||||
current = cache->most_recent_entry;
|
* or if hashing has just failed
|
||||||
while (current
|
*/
|
||||||
&& compare(current, wanted)) {
|
current = cache->most_recent_entry;
|
||||||
current = current->next;
|
while (current && compare(current, wanted))
|
||||||
}
|
{
|
||||||
}
|
current = current->next;
|
||||||
if (current) {
|
}
|
||||||
previous = current->previous;
|
}
|
||||||
cache->hits++;
|
if (current)
|
||||||
if (previous) {
|
{
|
||||||
/*
|
previous = current->previous;
|
||||||
* found and not at head of list, unlink from current
|
cache->hits++;
|
||||||
* position and relink as head of list
|
if (previous)
|
||||||
*/
|
{
|
||||||
previous->next = current->next;
|
/*
|
||||||
if (current->next)
|
* found and not at head of list, unlink from current
|
||||||
current->next->previous
|
* position and relink as head of list
|
||||||
= current->previous;
|
*/
|
||||||
else
|
previous->next = current->next;
|
||||||
cache->oldest_entry
|
if (current->next)
|
||||||
= current->previous;
|
current->next->previous = current->previous;
|
||||||
current->next = cache->most_recent_entry;
|
else cache->oldest_entry = current->previous;
|
||||||
current->previous
|
current->next = cache->most_recent_entry;
|
||||||
= (struct CACHED_GENERIC*)NULL;
|
current->previous = (struct CACHED_GENERIC*) NULL;
|
||||||
cache->most_recent_entry->previous = current;
|
cache->most_recent_entry->previous = current;
|
||||||
cache->most_recent_entry = current;
|
cache->most_recent_entry = current;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
cache->reads++;
|
cache->reads++;
|
||||||
}
|
}
|
||||||
return (current);
|
return (current);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -209,121 +220,125 @@ struct CACHED_GENERIC *ntfs_fetch_cache(struct CACHE_HEADER *cache,
|
|||||||
* returns the cache entry or NULL if not possible
|
* returns the cache entry or NULL if not possible
|
||||||
*/
|
*/
|
||||||
|
|
||||||
struct CACHED_GENERIC *ntfs_enter_cache(struct CACHE_HEADER *cache,
|
struct CACHED_GENERIC *ntfs_enter_cache(struct CACHE_HEADER *cache, const struct CACHED_GENERIC *item,
|
||||||
const struct CACHED_GENERIC *item,
|
cache_compare compare)
|
||||||
cache_compare compare)
|
|
||||||
{
|
{
|
||||||
struct CACHED_GENERIC *current;
|
struct CACHED_GENERIC *current;
|
||||||
struct CACHED_GENERIC *before;
|
struct CACHED_GENERIC *before;
|
||||||
struct HASH_ENTRY *link;
|
struct HASH_ENTRY *link;
|
||||||
int h;
|
int h;
|
||||||
|
|
||||||
current = (struct CACHED_GENERIC*)NULL;
|
current = (struct CACHED_GENERIC*) NULL;
|
||||||
if (cache) {
|
if (cache)
|
||||||
if (cache->dohash) {
|
{
|
||||||
/*
|
if (cache->dohash)
|
||||||
* When possible, use the hash table to
|
{
|
||||||
* find out whether the entry if present
|
/*
|
||||||
*/
|
* When possible, use the hash table to
|
||||||
h = cache->dohash(item);
|
* find out whether the entry if present
|
||||||
link = cache->first_hash[h];
|
*/
|
||||||
while (link && compare(link->entry, item))
|
h = cache->dohash(item);
|
||||||
link = link->next;
|
link = cache->first_hash[h];
|
||||||
if (link) {
|
while (link && compare(link->entry, item))
|
||||||
current = link->entry;
|
link = link->next;
|
||||||
}
|
if (link)
|
||||||
}
|
{
|
||||||
if (!cache->dohash) {
|
current = link->entry;
|
||||||
/*
|
}
|
||||||
* Search sequentially in LRU list to locate the end,
|
}
|
||||||
* and find out whether the entry is already in list
|
if (!cache->dohash)
|
||||||
* As we normally go to the end, no statistics is
|
{
|
||||||
* kept.
|
/*
|
||||||
*/
|
* Search sequentially in LRU list to locate the end,
|
||||||
current = cache->most_recent_entry;
|
* and find out whether the entry is already in list
|
||||||
while (current
|
* As we normally go to the end, no statistics is
|
||||||
&& compare(current, item)) {
|
* kept.
|
||||||
current = current->next;
|
*/
|
||||||
}
|
current = cache->most_recent_entry;
|
||||||
}
|
while (current && compare(current, item))
|
||||||
|
{
|
||||||
|
current = current->next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (!current) {
|
if (!current)
|
||||||
/*
|
{
|
||||||
* Not in list, get a free entry or reuse the
|
/*
|
||||||
* last entry, and relink as head of list
|
* Not in list, get a free entry or reuse the
|
||||||
* Note : we assume at least three entries, so
|
* last entry, and relink as head of list
|
||||||
* before, previous and first are different when
|
* Note : we assume at least three entries, so
|
||||||
* an entry is reused.
|
* before, previous and first are different when
|
||||||
*/
|
* an entry is reused.
|
||||||
|
*/
|
||||||
|
|
||||||
if (cache->free_entry) {
|
if (cache->free_entry)
|
||||||
current = cache->free_entry;
|
{
|
||||||
cache->free_entry = cache->free_entry->next;
|
current = cache->free_entry;
|
||||||
if (item->varsize) {
|
cache->free_entry = cache->free_entry->next;
|
||||||
current->variable = ntfs_malloc(
|
if (item->varsize)
|
||||||
item->varsize);
|
{
|
||||||
} else
|
current->variable = ntfs_malloc(item->varsize);
|
||||||
current->variable = (void*)NULL;
|
}
|
||||||
current->varsize = item->varsize;
|
else current->variable = (void*) NULL;
|
||||||
if (!cache->oldest_entry)
|
current->varsize = item->varsize;
|
||||||
cache->oldest_entry = current;
|
if (!cache->oldest_entry) cache->oldest_entry = current;
|
||||||
} else {
|
}
|
||||||
/* reusing the oldest entry */
|
else
|
||||||
current = cache->oldest_entry;
|
{
|
||||||
before = current->previous;
|
/* reusing the oldest entry */
|
||||||
before->next = (struct CACHED_GENERIC*)NULL;
|
current = cache->oldest_entry;
|
||||||
if (cache->dohash)
|
before = current->previous;
|
||||||
drophashindex(cache,current,
|
before->next = (struct CACHED_GENERIC*) NULL;
|
||||||
cache->dohash(current));
|
if (cache->dohash) drophashindex(cache, current, cache->dohash(current));
|
||||||
if (cache->dofree)
|
if (cache->dofree) cache->dofree(current);
|
||||||
cache->dofree(current);
|
cache->oldest_entry = current->previous;
|
||||||
cache->oldest_entry = current->previous;
|
if (item->varsize)
|
||||||
if (item->varsize) {
|
{
|
||||||
if (current->varsize)
|
if (current->varsize)
|
||||||
current->variable = realloc(
|
current->variable = realloc(current->variable, item->varsize);
|
||||||
current->variable,
|
else current->variable = ntfs_malloc(item->varsize);
|
||||||
item->varsize);
|
}
|
||||||
else
|
else
|
||||||
current->variable = ntfs_malloc(
|
{
|
||||||
item->varsize);
|
if (current->varsize) free(current->variable);
|
||||||
} else {
|
current->variable = (void*) NULL;
|
||||||
if (current->varsize)
|
}
|
||||||
free(current->variable);
|
current->varsize = item->varsize;
|
||||||
current->variable = (void*)NULL;
|
}
|
||||||
}
|
current->next = cache->most_recent_entry;
|
||||||
current->varsize = item->varsize;
|
current->previous = (struct CACHED_GENERIC*) NULL;
|
||||||
}
|
if (cache->most_recent_entry) cache->most_recent_entry->previous = current;
|
||||||
current->next = cache->most_recent_entry;
|
cache->most_recent_entry = current;
|
||||||
current->previous = (struct CACHED_GENERIC*)NULL;
|
memcpy(current->fixed, item->fixed, cache->fixed_size);
|
||||||
if (cache->most_recent_entry)
|
if (item->varsize)
|
||||||
cache->most_recent_entry->previous = current;
|
{
|
||||||
cache->most_recent_entry = current;
|
if (current->variable)
|
||||||
memcpy(current->fixed, item->fixed, cache->fixed_size);
|
{
|
||||||
if (item->varsize) {
|
memcpy(current->variable, item->variable, item->varsize);
|
||||||
if (current->variable) {
|
}
|
||||||
memcpy(current->variable,
|
else
|
||||||
item->variable, item->varsize);
|
{
|
||||||
} else {
|
/*
|
||||||
/*
|
* no more memory for variable part
|
||||||
* no more memory for variable part
|
* recycle entry in free list
|
||||||
* recycle entry in free list
|
* not an error, just uncacheable
|
||||||
* not an error, just uncacheable
|
*/
|
||||||
*/
|
cache->most_recent_entry = current->next;
|
||||||
cache->most_recent_entry = current->next;
|
current->next = cache->free_entry;
|
||||||
current->next = cache->free_entry;
|
cache->free_entry = current;
|
||||||
cache->free_entry = current;
|
current = (struct CACHED_GENERIC*) NULL;
|
||||||
current = (struct CACHED_GENERIC*)NULL;
|
}
|
||||||
}
|
}
|
||||||
} else {
|
else
|
||||||
current->variable = (void*)NULL;
|
{
|
||||||
current->varsize = 0;
|
current->variable = (void*) NULL;
|
||||||
}
|
current->varsize = 0;
|
||||||
if (cache->dohash && current)
|
}
|
||||||
inserthashindex(cache,current);
|
if (cache->dohash && current) inserthashindex(cache, current);
|
||||||
}
|
}
|
||||||
cache->writes++;
|
cache->writes++;
|
||||||
}
|
}
|
||||||
return (current);
|
return (current);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -332,32 +347,26 @@ struct CACHED_GENERIC *ntfs_enter_cache(struct CACHE_HEADER *cache,
|
|||||||
* A specific function may be called for entry deletion
|
* A specific function may be called for entry deletion
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static void do_invalidate(struct CACHE_HEADER *cache,
|
static void do_invalidate(struct CACHE_HEADER *cache, struct CACHED_GENERIC *current, int flags)
|
||||||
struct CACHED_GENERIC *current, int flags)
|
|
||||||
{
|
{
|
||||||
struct CACHED_GENERIC *previous;
|
struct CACHED_GENERIC *previous;
|
||||||
|
|
||||||
previous = current->previous;
|
|
||||||
if ((flags & CACHE_FREE) && cache->dofree)
|
|
||||||
cache->dofree(current);
|
|
||||||
/*
|
|
||||||
* Relink into free list
|
|
||||||
*/
|
|
||||||
if (current->next)
|
|
||||||
current->next->previous = current->previous;
|
|
||||||
else
|
|
||||||
cache->oldest_entry = current->previous;
|
|
||||||
if (previous)
|
|
||||||
previous->next = current->next;
|
|
||||||
else
|
|
||||||
cache->most_recent_entry = current->next;
|
|
||||||
current->next = cache->free_entry;
|
|
||||||
cache->free_entry = current;
|
|
||||||
if (current->variable)
|
|
||||||
free(current->variable);
|
|
||||||
current->varsize = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
previous = current->previous;
|
||||||
|
if ((flags & CACHE_FREE) && cache->dofree) cache->dofree(current);
|
||||||
|
/*
|
||||||
|
* Relink into free list
|
||||||
|
*/
|
||||||
|
if (current->next)
|
||||||
|
current->next->previous = current->previous;
|
||||||
|
else cache->oldest_entry = current->previous;
|
||||||
|
if (previous)
|
||||||
|
previous->next = current->next;
|
||||||
|
else cache->most_recent_entry = current->next;
|
||||||
|
current->next = cache->free_entry;
|
||||||
|
cache->free_entry = current;
|
||||||
|
if (current->variable) free(current->variable);
|
||||||
|
current->varsize = 0;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Invalidate entries in cache
|
* Invalidate entries in cache
|
||||||
@ -371,80 +380,85 @@ static void do_invalidate(struct CACHE_HEADER *cache,
|
|||||||
* supposed to be found.
|
* supposed to be found.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
int ntfs_invalidate_cache(struct CACHE_HEADER *cache,
|
int ntfs_invalidate_cache(struct CACHE_HEADER *cache, const struct CACHED_GENERIC *item, cache_compare compare,
|
||||||
const struct CACHED_GENERIC *item, cache_compare compare,
|
int flags)
|
||||||
int flags)
|
|
||||||
{
|
{
|
||||||
struct CACHED_GENERIC *current;
|
struct CACHED_GENERIC *current;
|
||||||
struct CACHED_GENERIC *previous;
|
struct CACHED_GENERIC *previous;
|
||||||
struct CACHED_GENERIC *next;
|
struct CACHED_GENERIC *next;
|
||||||
struct HASH_ENTRY *link;
|
struct HASH_ENTRY *link;
|
||||||
int count;
|
int count;
|
||||||
int h;
|
int h;
|
||||||
|
|
||||||
current = (struct CACHED_GENERIC*)NULL;
|
current = (struct CACHED_GENERIC*) NULL;
|
||||||
count = 0;
|
count = 0;
|
||||||
if (cache) {
|
if (cache)
|
||||||
if (!(flags & CACHE_NOHASH) && cache->dohash) {
|
{
|
||||||
/*
|
if (!(flags & CACHE_NOHASH) && cache->dohash)
|
||||||
* When possible, use the hash table to
|
{
|
||||||
* find out whether the entry if present
|
/*
|
||||||
*/
|
* When possible, use the hash table to
|
||||||
h = cache->dohash(item);
|
* find out whether the entry if present
|
||||||
link = cache->first_hash[h];
|
*/
|
||||||
while (link) {
|
h = cache->dohash(item);
|
||||||
if (compare(link->entry, item))
|
link = cache->first_hash[h];
|
||||||
link = link->next;
|
while (link)
|
||||||
else {
|
{
|
||||||
current = link->entry;
|
if (compare(link->entry, item))
|
||||||
link = link->next;
|
link = link->next;
|
||||||
if (current) {
|
else
|
||||||
drophashindex(cache,current,h);
|
{
|
||||||
do_invalidate(cache,
|
current = link->entry;
|
||||||
current,flags);
|
link = link->next;
|
||||||
count++;
|
if (current)
|
||||||
}
|
{
|
||||||
}
|
drophashindex(cache, current, h);
|
||||||
}
|
do_invalidate(cache, current, flags);
|
||||||
}
|
count++;
|
||||||
if ((flags & CACHE_NOHASH) || !cache->dohash) {
|
}
|
||||||
/*
|
}
|
||||||
* Search sequentially in LRU list
|
}
|
||||||
*/
|
}
|
||||||
current = cache->most_recent_entry;
|
if ((flags & CACHE_NOHASH) || !cache->dohash)
|
||||||
previous = (struct CACHED_GENERIC*)NULL;
|
{
|
||||||
while (current) {
|
/*
|
||||||
if (!compare(current, item)) {
|
* Search sequentially in LRU list
|
||||||
next = current->next;
|
*/
|
||||||
if (cache->dohash)
|
current = cache->most_recent_entry;
|
||||||
drophashindex(cache,current,
|
previous = (struct CACHED_GENERIC*) NULL;
|
||||||
cache->dohash(current));
|
while (current)
|
||||||
do_invalidate(cache,current,flags);
|
{
|
||||||
current = next;
|
if (!compare(current, item))
|
||||||
count++;
|
{
|
||||||
} else {
|
next = current->next;
|
||||||
previous = current;
|
if (cache->dohash) drophashindex(cache, current, cache->dohash(current));
|
||||||
current = current->next;
|
do_invalidate(cache, current, flags);
|
||||||
}
|
current = next;
|
||||||
}
|
count++;
|
||||||
}
|
}
|
||||||
}
|
else
|
||||||
return (count);
|
{
|
||||||
|
previous = current;
|
||||||
|
current = current->next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return (count);
|
||||||
}
|
}
|
||||||
|
|
||||||
int ntfs_remove_cache(struct CACHE_HEADER *cache,
|
int ntfs_remove_cache(struct CACHE_HEADER *cache, struct CACHED_GENERIC *item, int flags)
|
||||||
struct CACHED_GENERIC *item, int flags)
|
|
||||||
{
|
{
|
||||||
int count;
|
int count;
|
||||||
|
|
||||||
count = 0;
|
count = 0;
|
||||||
if (cache) {
|
if (cache)
|
||||||
if (cache->dohash)
|
{
|
||||||
drophashindex(cache,item,cache->dohash(item));
|
if (cache->dohash) drophashindex(cache, item, cache->dohash(item));
|
||||||
do_invalidate(cache,item,flags);
|
do_invalidate(cache, item, flags);
|
||||||
count++;
|
count++;
|
||||||
}
|
}
|
||||||
return (count);
|
return (count);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -453,17 +467,17 @@ int ntfs_remove_cache(struct CACHE_HEADER *cache,
|
|||||||
|
|
||||||
static void ntfs_free_cache(struct CACHE_HEADER *cache)
|
static void ntfs_free_cache(struct CACHE_HEADER *cache)
|
||||||
{
|
{
|
||||||
struct CACHED_GENERIC *entry;
|
struct CACHED_GENERIC *entry;
|
||||||
|
|
||||||
if (cache) {
|
if (cache)
|
||||||
for (entry=cache->most_recent_entry; entry; entry=entry->next) {
|
{
|
||||||
if (cache->dofree)
|
for (entry = cache->most_recent_entry; entry; entry = entry->next)
|
||||||
cache->dofree(entry);
|
{
|
||||||
if (entry->variable)
|
if (cache->dofree) cache->dofree(entry);
|
||||||
free(entry->variable);
|
if (entry->variable) free(entry->variable);
|
||||||
}
|
}
|
||||||
free(cache);
|
free(cache);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -472,82 +486,87 @@ static void ntfs_free_cache(struct CACHE_HEADER *cache)
|
|||||||
* Returns the cache header, or NULL if the cache could not be created
|
* Returns the cache header, or NULL if the cache could not be created
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static struct CACHE_HEADER *ntfs_create_cache(const char *name,
|
static struct CACHE_HEADER *ntfs_create_cache(const char *name, cache_free dofree, cache_hash dohash,
|
||||||
cache_free dofree, cache_hash dohash,
|
int full_item_size, int item_count, int max_hash)
|
||||||
int full_item_size,
|
|
||||||
int item_count, int max_hash)
|
|
||||||
{
|
{
|
||||||
struct CACHE_HEADER *cache;
|
struct CACHE_HEADER *cache;
|
||||||
struct CACHED_GENERIC *pc;
|
struct CACHED_GENERIC *pc;
|
||||||
struct CACHED_GENERIC *qc;
|
struct CACHED_GENERIC *qc;
|
||||||
struct HASH_ENTRY *ph;
|
struct HASH_ENTRY *ph;
|
||||||
struct HASH_ENTRY *qh;
|
struct HASH_ENTRY *qh;
|
||||||
struct HASH_ENTRY **px;
|
struct HASH_ENTRY **px;
|
||||||
size_t size;
|
size_t size;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
size = sizeof(struct CACHE_HEADER) + item_count*full_item_size;
|
size = sizeof(struct CACHE_HEADER) + item_count * full_item_size;
|
||||||
if (max_hash)
|
if (max_hash) size += item_count * sizeof(struct HASH_ENTRY) + max_hash * sizeof(struct HASH_ENTRY*);
|
||||||
size += item_count*sizeof(struct HASH_ENTRY)
|
cache = (struct CACHE_HEADER*) ntfs_malloc(size);
|
||||||
+ max_hash*sizeof(struct HASH_ENTRY*);
|
if (cache)
|
||||||
cache = (struct CACHE_HEADER*)ntfs_malloc(size);
|
{
|
||||||
if (cache) {
|
/* header */
|
||||||
/* header */
|
cache->name = name;
|
||||||
cache->name = name;
|
cache->dofree = dofree;
|
||||||
cache->dofree = dofree;
|
if (dohash && max_hash)
|
||||||
if (dohash && max_hash) {
|
{
|
||||||
cache->dohash = dohash;
|
cache->dohash = dohash;
|
||||||
cache->max_hash = max_hash;
|
cache->max_hash = max_hash;
|
||||||
} else {
|
}
|
||||||
cache->dohash = (cache_hash)NULL;
|
else
|
||||||
cache->max_hash = 0;
|
{
|
||||||
}
|
cache->dohash = (cache_hash) NULL;
|
||||||
cache->fixed_size = full_item_size - sizeof(struct CACHED_GENERIC);
|
cache->max_hash = 0;
|
||||||
cache->reads = 0;
|
}
|
||||||
cache->writes = 0;
|
cache->fixed_size = full_item_size - sizeof(struct CACHED_GENERIC);
|
||||||
cache->hits = 0;
|
cache->reads = 0;
|
||||||
/* chain the data entries, and mark an invalid entry */
|
cache->writes = 0;
|
||||||
cache->most_recent_entry = (struct CACHED_GENERIC*)NULL;
|
cache->hits = 0;
|
||||||
cache->oldest_entry = (struct CACHED_GENERIC*)NULL;
|
/* chain the data entries, and mark an invalid entry */
|
||||||
cache->free_entry = &cache->entry[0];
|
cache->most_recent_entry = (struct CACHED_GENERIC*) NULL;
|
||||||
pc = &cache->entry[0];
|
cache->oldest_entry = (struct CACHED_GENERIC*) NULL;
|
||||||
for (i=0; i<(item_count - 1); i++) {
|
cache->free_entry = &cache->entry[0];
|
||||||
qc = (struct CACHED_GENERIC*)((char*)pc
|
pc = &cache->entry[0];
|
||||||
+ full_item_size);
|
for (i = 0; i < (item_count - 1); i++)
|
||||||
pc->next = qc;
|
{
|
||||||
pc->variable = (void*)NULL;
|
qc = (struct CACHED_GENERIC*) ((char*) pc + full_item_size);
|
||||||
pc->varsize = 0;
|
pc->next = qc;
|
||||||
pc = qc;
|
pc->variable = (void*) NULL;
|
||||||
}
|
pc->varsize = 0;
|
||||||
/* special for the last entry */
|
pc = qc;
|
||||||
pc->next = (struct CACHED_GENERIC*)NULL;
|
}
|
||||||
pc->variable = (void*)NULL;
|
/* special for the last entry */
|
||||||
pc->varsize = 0;
|
pc->next = (struct CACHED_GENERIC*) NULL;
|
||||||
|
pc->variable = (void*) NULL;
|
||||||
|
pc->varsize = 0;
|
||||||
|
|
||||||
if (max_hash) {
|
if (max_hash)
|
||||||
/* chain the hash entries */
|
{
|
||||||
ph = (struct HASH_ENTRY*)(((char*)pc) + full_item_size);
|
/* chain the hash entries */
|
||||||
cache->free_hash = ph;
|
ph = (struct HASH_ENTRY*) (((char*) pc) + full_item_size);
|
||||||
for (i=0; i<(item_count - 1); i++) {
|
cache->free_hash = ph;
|
||||||
qh = &ph[1];
|
for (i = 0; i < (item_count - 1); i++)
|
||||||
ph->next = qh;
|
{
|
||||||
ph = qh;
|
qh = &ph[1];
|
||||||
}
|
ph->next = qh;
|
||||||
/* special for the last entry */
|
ph = qh;
|
||||||
if (item_count) {
|
}
|
||||||
ph->next = (struct HASH_ENTRY*)NULL;
|
/* special for the last entry */
|
||||||
}
|
if (item_count)
|
||||||
/* create and initialize the hash indexes */
|
{
|
||||||
px = (struct HASH_ENTRY**)&ph[1];
|
ph->next = (struct HASH_ENTRY*) NULL;
|
||||||
cache->first_hash = px;
|
}
|
||||||
for (i=0; i<max_hash; i++)
|
/* create and initialize the hash indexes */
|
||||||
px[i] = (struct HASH_ENTRY*)NULL;
|
px = (struct HASH_ENTRY**) &ph[1];
|
||||||
} else {
|
cache->first_hash = px;
|
||||||
cache->free_hash = (struct HASH_ENTRY*)NULL;
|
for (i = 0; i < max_hash; i++)
|
||||||
cache->first_hash = (struct HASH_ENTRY**)NULL;
|
px[i] = (struct HASH_ENTRY*) NULL;
|
||||||
}
|
}
|
||||||
}
|
else
|
||||||
return (cache);
|
{
|
||||||
|
cache->free_hash = (struct HASH_ENTRY*) NULL;
|
||||||
|
cache->first_hash = (struct HASH_ENTRY**) NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return (cache);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -560,30 +579,25 @@ static struct CACHE_HEADER *ntfs_create_cache(const char *name,
|
|||||||
void ntfs_create_lru_caches(ntfs_volume *vol)
|
void ntfs_create_lru_caches(ntfs_volume *vol)
|
||||||
{
|
{
|
||||||
#if CACHE_INODE_SIZE
|
#if CACHE_INODE_SIZE
|
||||||
/* inode cache */
|
/* inode cache */
|
||||||
vol->xinode_cache = ntfs_create_cache("inode",(cache_free)NULL,
|
vol->xinode_cache = ntfs_create_cache("inode", (cache_free) NULL, ntfs_dir_inode_hash, sizeof(struct CACHED_INODE),
|
||||||
ntfs_dir_inode_hash, sizeof(struct CACHED_INODE),
|
CACHE_INODE_SIZE, 2 * CACHE_INODE_SIZE);
|
||||||
CACHE_INODE_SIZE, 2*CACHE_INODE_SIZE);
|
|
||||||
#endif
|
#endif
|
||||||
#if CACHE_NIDATA_SIZE
|
#if CACHE_NIDATA_SIZE
|
||||||
/* idata cache */
|
/* idata cache */
|
||||||
vol->nidata_cache = ntfs_create_cache("nidata",
|
vol->nidata_cache = ntfs_create_cache("nidata", ntfs_inode_nidata_free, ntfs_inode_nidata_hash,
|
||||||
ntfs_inode_nidata_free, ntfs_inode_nidata_hash,
|
sizeof(struct CACHED_NIDATA), CACHE_NIDATA_SIZE, 2 * CACHE_NIDATA_SIZE);
|
||||||
sizeof(struct CACHED_NIDATA),
|
|
||||||
CACHE_NIDATA_SIZE, 2*CACHE_NIDATA_SIZE);
|
|
||||||
#endif
|
#endif
|
||||||
#if CACHE_LOOKUP_SIZE
|
#if CACHE_LOOKUP_SIZE
|
||||||
/* lookup cache */
|
/* lookup cache */
|
||||||
vol->lookup_cache = ntfs_create_cache("lookup",
|
vol->lookup_cache = ntfs_create_cache("lookup", (cache_free) NULL, ntfs_dir_lookup_hash,
|
||||||
(cache_free)NULL, ntfs_dir_lookup_hash,
|
sizeof(struct CACHED_LOOKUP), CACHE_LOOKUP_SIZE, 2 * CACHE_LOOKUP_SIZE);
|
||||||
sizeof(struct CACHED_LOOKUP),
|
|
||||||
CACHE_LOOKUP_SIZE, 2*CACHE_LOOKUP_SIZE);
|
|
||||||
#endif
|
#endif
|
||||||
vol->securid_cache = ntfs_create_cache("securid",(cache_free)NULL,
|
vol->securid_cache = ntfs_create_cache("securid", (cache_free) NULL, (cache_hash) NULL,
|
||||||
(cache_hash)NULL,sizeof(struct CACHED_SECURID), CACHE_SECURID_SIZE, 0);
|
sizeof(struct CACHED_SECURID), CACHE_SECURID_SIZE, 0);
|
||||||
#if CACHE_LEGACY_SIZE
|
#if CACHE_LEGACY_SIZE
|
||||||
vol->legacy_cache = ntfs_create_cache("legacy",(cache_free)NULL,
|
vol->legacy_cache = ntfs_create_cache("legacy", (cache_free) NULL, (cache_hash) NULL,
|
||||||
(cache_hash)NULL, sizeof(struct CACHED_PERMISSIONS_LEGACY), CACHE_LEGACY_SIZE, 0);
|
sizeof(struct CACHED_PERMISSIONS_LEGACY), CACHE_LEGACY_SIZE, 0);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -594,16 +608,16 @@ void ntfs_create_lru_caches(ntfs_volume *vol)
|
|||||||
void ntfs_free_lru_caches(ntfs_volume *vol)
|
void ntfs_free_lru_caches(ntfs_volume *vol)
|
||||||
{
|
{
|
||||||
#if CACHE_INODE_SIZE
|
#if CACHE_INODE_SIZE
|
||||||
ntfs_free_cache(vol->xinode_cache);
|
ntfs_free_cache(vol->xinode_cache);
|
||||||
#endif
|
#endif
|
||||||
#if CACHE_NIDATA_SIZE
|
#if CACHE_NIDATA_SIZE
|
||||||
ntfs_free_cache(vol->nidata_cache);
|
ntfs_free_cache(vol->nidata_cache);
|
||||||
#endif
|
#endif
|
||||||
#if CACHE_LOOKUP_SIZE
|
#if CACHE_LOOKUP_SIZE
|
||||||
ntfs_free_cache(vol->lookup_cache);
|
ntfs_free_cache(vol->lookup_cache);
|
||||||
#endif
|
#endif
|
||||||
ntfs_free_cache(vol->securid_cache);
|
ntfs_free_cache(vol->securid_cache);
|
||||||
#if CACHE_LEGACY_SIZE
|
#if CACHE_LEGACY_SIZE
|
||||||
ntfs_free_cache(vol->legacy_cache);
|
ntfs_free_cache(vol->legacy_cache);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
@ -24,93 +24,95 @@
|
|||||||
|
|
||||||
#include "volume.h"
|
#include "volume.h"
|
||||||
|
|
||||||
struct CACHED_GENERIC {
|
struct CACHED_GENERIC
|
||||||
struct CACHED_GENERIC *next;
|
{
|
||||||
struct CACHED_GENERIC *previous;
|
struct CACHED_GENERIC *next;
|
||||||
void *variable;
|
struct CACHED_GENERIC *previous;
|
||||||
size_t varsize;
|
void *variable;
|
||||||
union {
|
size_t varsize;
|
||||||
/* force alignment for pointers and u64 */
|
union
|
||||||
u64 u64align;
|
{
|
||||||
void *ptralign;
|
/* force alignment for pointers and u64 */
|
||||||
} fixed[0];
|
u64 u64align;
|
||||||
} ;
|
void *ptralign;
|
||||||
|
} fixed[0];
|
||||||
|
};
|
||||||
|
|
||||||
struct CACHED_INODE {
|
struct CACHED_INODE
|
||||||
struct CACHED_INODE *next;
|
{
|
||||||
struct CACHED_INODE *previous;
|
struct CACHED_INODE *next;
|
||||||
const char *pathname;
|
struct CACHED_INODE *previous;
|
||||||
size_t varsize;
|
const char *pathname;
|
||||||
/* above fields must match "struct CACHED_GENERIC" */
|
size_t varsize;
|
||||||
u64 inum;
|
/* above fields must match "struct CACHED_GENERIC" */
|
||||||
} ;
|
u64 inum;
|
||||||
|
};
|
||||||
|
|
||||||
struct CACHED_NIDATA {
|
struct CACHED_NIDATA
|
||||||
struct CACHED_NIDATA *next;
|
{
|
||||||
struct CACHED_NIDATA *previous;
|
struct CACHED_NIDATA *next;
|
||||||
const char *pathname; /* not used */
|
struct CACHED_NIDATA *previous;
|
||||||
size_t varsize; /* not used */
|
const char *pathname; /* not used */
|
||||||
/* above fields must match "struct CACHED_GENERIC" */
|
size_t varsize; /* not used */
|
||||||
u64 inum;
|
/* above fields must match "struct CACHED_GENERIC" */
|
||||||
ntfs_inode *ni;
|
u64 inum;
|
||||||
} ;
|
ntfs_inode *ni;
|
||||||
|
};
|
||||||
|
|
||||||
struct CACHED_LOOKUP {
|
struct CACHED_LOOKUP
|
||||||
struct CACHED_LOOKUP *next;
|
{
|
||||||
struct CACHED_LOOKUP *previous;
|
struct CACHED_LOOKUP *next;
|
||||||
const char *name;
|
struct CACHED_LOOKUP *previous;
|
||||||
size_t namesize;
|
const char *name;
|
||||||
/* above fields must match "struct CACHED_GENERIC" */
|
size_t namesize;
|
||||||
u64 parent;
|
/* above fields must match "struct CACHED_GENERIC" */
|
||||||
u64 inum;
|
u64 parent;
|
||||||
} ;
|
u64 inum;
|
||||||
|
};
|
||||||
|
|
||||||
enum {
|
enum
|
||||||
CACHE_FREE = 1,
|
{
|
||||||
CACHE_NOHASH = 2
|
CACHE_FREE = 1, CACHE_NOHASH = 2
|
||||||
} ;
|
};
|
||||||
|
|
||||||
typedef int (*cache_compare)(const struct CACHED_GENERIC *cached,
|
typedef int (*cache_compare)(const struct CACHED_GENERIC *cached, const struct CACHED_GENERIC *item);
|
||||||
const struct CACHED_GENERIC *item);
|
|
||||||
typedef void (*cache_free)(const struct CACHED_GENERIC *cached);
|
typedef void (*cache_free)(const struct CACHED_GENERIC *cached);
|
||||||
typedef int (*cache_hash)(const struct CACHED_GENERIC *cached);
|
typedef int (*cache_hash)(const struct CACHED_GENERIC *cached);
|
||||||
|
|
||||||
struct HASH_ENTRY {
|
struct HASH_ENTRY
|
||||||
struct HASH_ENTRY *next;
|
{
|
||||||
struct CACHED_GENERIC *entry;
|
struct HASH_ENTRY *next;
|
||||||
} ;
|
struct CACHED_GENERIC *entry;
|
||||||
|
};
|
||||||
|
|
||||||
struct CACHE_HEADER {
|
struct CACHE_HEADER
|
||||||
const char *name;
|
{
|
||||||
struct CACHED_GENERIC *most_recent_entry;
|
const char *name;
|
||||||
struct CACHED_GENERIC *oldest_entry;
|
struct CACHED_GENERIC *most_recent_entry;
|
||||||
struct CACHED_GENERIC *free_entry;
|
struct CACHED_GENERIC *oldest_entry;
|
||||||
struct HASH_ENTRY *free_hash;
|
struct CACHED_GENERIC *free_entry;
|
||||||
struct HASH_ENTRY **first_hash;
|
struct HASH_ENTRY *free_hash;
|
||||||
cache_free dofree;
|
struct HASH_ENTRY **first_hash;
|
||||||
cache_hash dohash;
|
cache_free dofree;
|
||||||
unsigned long reads;
|
cache_hash dohash;
|
||||||
unsigned long writes;
|
unsigned long reads;
|
||||||
unsigned long hits;
|
unsigned long writes;
|
||||||
int fixed_size;
|
unsigned long hits;
|
||||||
int max_hash;
|
int fixed_size;
|
||||||
struct CACHED_GENERIC entry[0];
|
int max_hash;
|
||||||
} ;
|
struct CACHED_GENERIC entry[0];
|
||||||
|
};
|
||||||
|
|
||||||
/* cast to generic, avoiding gcc warnings */
|
/* cast to generic, avoiding gcc warnings */
|
||||||
#define GENERIC(pstr) ((const struct CACHED_GENERIC*)(const void*)(pstr))
|
#define GENERIC(pstr) ((const struct CACHED_GENERIC*)(const void*)(pstr))
|
||||||
|
|
||||||
struct CACHED_GENERIC *ntfs_fetch_cache(struct CACHE_HEADER *cache,
|
struct CACHED_GENERIC *ntfs_fetch_cache(struct CACHE_HEADER *cache, const struct CACHED_GENERIC *wanted,
|
||||||
const struct CACHED_GENERIC *wanted,
|
cache_compare compare);
|
||||||
cache_compare compare);
|
struct CACHED_GENERIC *ntfs_enter_cache(struct CACHE_HEADER *cache, const struct CACHED_GENERIC *item,
|
||||||
struct CACHED_GENERIC *ntfs_enter_cache(struct CACHE_HEADER *cache,
|
cache_compare compare);
|
||||||
const struct CACHED_GENERIC *item,
|
int ntfs_invalidate_cache(struct CACHE_HEADER *cache, const struct CACHED_GENERIC *item, cache_compare compare,
|
||||||
cache_compare compare);
|
int flags);
|
||||||
int ntfs_invalidate_cache(struct CACHE_HEADER *cache,
|
int ntfs_remove_cache(struct CACHE_HEADER *cache, struct CACHED_GENERIC *item, int flags);
|
||||||
const struct CACHED_GENERIC *item,
|
|
||||||
cache_compare compare, int flags);
|
|
||||||
int ntfs_remove_cache(struct CACHE_HEADER *cache,
|
|
||||||
struct CACHED_GENERIC *item, int flags);
|
|
||||||
|
|
||||||
void ntfs_create_lru_caches(ntfs_volume *vol);
|
void ntfs_create_lru_caches(ntfs_volume *vol);
|
||||||
void ntfs_free_lru_caches(ntfs_volume *vol);
|
void ntfs_free_lru_caches(ntfs_volume *vol);
|
||||||
|
@ -16,13 +16,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -33,7 +33,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <ogc/lwp_watchdog.h>
|
#include <ogc/lwp_watchdog.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
@ -45,330 +45,384 @@
|
|||||||
|
|
||||||
#define CACHE_FREE UINT_MAX
|
#define CACHE_FREE UINT_MAX
|
||||||
|
|
||||||
NTFS_CACHE* _NTFS_cache_constructor (unsigned int numberOfPages, unsigned int sectorsPerPage, const DISC_INTERFACE* discInterface, sec_t endOfPartition, sec_t sectorSize) {
|
NTFS_CACHE* _NTFS_cache_constructor(unsigned int numberOfPages, unsigned int sectorsPerPage,
|
||||||
NTFS_CACHE* cache;
|
const DISC_INTERFACE* discInterface, sec_t endOfPartition, sec_t sectorSize)
|
||||||
unsigned int i;
|
{
|
||||||
NTFS_CACHE_ENTRY* cacheEntries;
|
NTFS_CACHE* cache;
|
||||||
|
unsigned int i;
|
||||||
|
NTFS_CACHE_ENTRY* cacheEntries;
|
||||||
|
|
||||||
if(numberOfPages==0 || sectorsPerPage==0) return NULL;
|
if (numberOfPages == 0 || sectorsPerPage == 0) return NULL;
|
||||||
|
|
||||||
if (numberOfPages < 4) {
|
if (numberOfPages < 4)
|
||||||
numberOfPages = 4;
|
{
|
||||||
}
|
numberOfPages = 4;
|
||||||
|
}
|
||||||
|
|
||||||
if (sectorsPerPage < 32) {
|
if (sectorsPerPage < 32)
|
||||||
sectorsPerPage = 32;
|
{
|
||||||
}
|
sectorsPerPage = 32;
|
||||||
|
}
|
||||||
|
|
||||||
cache = (NTFS_CACHE*) ntfs_alloc (sizeof(NTFS_CACHE));
|
cache = (NTFS_CACHE*) ntfs_alloc(sizeof(NTFS_CACHE));
|
||||||
if (cache == NULL) {
|
if (cache == NULL)
|
||||||
return NULL;
|
{
|
||||||
}
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
cache->disc = discInterface;
|
cache->disc = discInterface;
|
||||||
cache->endOfPartition = endOfPartition;
|
cache->endOfPartition = endOfPartition;
|
||||||
cache->numberOfPages = numberOfPages;
|
cache->numberOfPages = numberOfPages;
|
||||||
cache->sectorsPerPage = sectorsPerPage;
|
cache->sectorsPerPage = sectorsPerPage;
|
||||||
cache->sectorSize = sectorSize;
|
cache->sectorSize = sectorSize;
|
||||||
|
|
||||||
|
cacheEntries = (NTFS_CACHE_ENTRY*) ntfs_alloc(sizeof(NTFS_CACHE_ENTRY) * numberOfPages);
|
||||||
|
if (cacheEntries == NULL)
|
||||||
|
{
|
||||||
|
ntfs_free(cache);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
cacheEntries = (NTFS_CACHE_ENTRY*) ntfs_alloc ( sizeof(NTFS_CACHE_ENTRY) * numberOfPages);
|
for (i = 0; i < numberOfPages; i++)
|
||||||
if (cacheEntries == NULL) {
|
{
|
||||||
ntfs_free (cache);
|
cacheEntries[i].sector = CACHE_FREE;
|
||||||
return NULL;
|
cacheEntries[i].count = 0;
|
||||||
}
|
cacheEntries[i].last_access = 0;
|
||||||
|
cacheEntries[i].dirty = false;
|
||||||
|
cacheEntries[i].cache = (uint8_t*) ntfs_align(sectorsPerPage * cache->sectorSize);
|
||||||
|
}
|
||||||
|
|
||||||
for (i = 0; i < numberOfPages; i++) {
|
cache->cacheEntries = cacheEntries;
|
||||||
cacheEntries[i].sector = CACHE_FREE;
|
|
||||||
cacheEntries[i].count = 0;
|
|
||||||
cacheEntries[i].last_access = 0;
|
|
||||||
cacheEntries[i].dirty = false;
|
|
||||||
cacheEntries[i].cache = (uint8_t*) ntfs_align ( sectorsPerPage * cache->sectorSize );
|
|
||||||
}
|
|
||||||
|
|
||||||
cache->cacheEntries = cacheEntries;
|
return cache;
|
||||||
|
|
||||||
return cache;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void _NTFS_cache_destructor (NTFS_CACHE* cache) {
|
void _NTFS_cache_destructor(NTFS_CACHE* cache)
|
||||||
unsigned int i;
|
{
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
if(cache==NULL) return;
|
if (cache == NULL) return;
|
||||||
|
|
||||||
// Clear out cache before destroying it
|
// Clear out cache before destroying it
|
||||||
_NTFS_cache_flush(cache);
|
_NTFS_cache_flush(cache);
|
||||||
|
|
||||||
// Free memory in reverse allocation order
|
// Free memory in reverse allocation order
|
||||||
for (i = 0; i < cache->numberOfPages; i++) {
|
for (i = 0; i < cache->numberOfPages; i++)
|
||||||
ntfs_free (cache->cacheEntries[i].cache);
|
{
|
||||||
}
|
ntfs_free(cache->cacheEntries[i].cache);
|
||||||
ntfs_free (cache->cacheEntries);
|
}
|
||||||
ntfs_free (cache);
|
ntfs_free(cache->cacheEntries);
|
||||||
|
ntfs_free(cache);
|
||||||
}
|
}
|
||||||
|
|
||||||
static u32 accessCounter = 0;
|
static u32 accessCounter = 0;
|
||||||
|
|
||||||
static u32 accessTime(){
|
static u32 accessTime()
|
||||||
accessCounter++;
|
|
||||||
return accessCounter;
|
|
||||||
}
|
|
||||||
|
|
||||||
static NTFS_CACHE_ENTRY* _NTFS_cache_getPage(NTFS_CACHE *cache,sec_t sector)
|
|
||||||
{
|
{
|
||||||
unsigned int i;
|
accessCounter++;
|
||||||
NTFS_CACHE_ENTRY* cacheEntries = cache->cacheEntries;
|
return accessCounter;
|
||||||
unsigned int numberOfPages = cache->numberOfPages;
|
|
||||||
unsigned int sectorsPerPage = cache->sectorsPerPage;
|
|
||||||
|
|
||||||
bool foundFree = false;
|
|
||||||
unsigned int oldUsed = 0;
|
|
||||||
unsigned int oldAccess = UINT_MAX;
|
|
||||||
|
|
||||||
for(i=0;i<numberOfPages;i++) {
|
|
||||||
if(sector>=cacheEntries[i].sector && sector<(cacheEntries[i].sector + cacheEntries[i].count)) {
|
|
||||||
cacheEntries[i].last_access = accessTime();
|
|
||||||
return &(cacheEntries[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
if(foundFree==false && (cacheEntries[i].sector==CACHE_FREE || cacheEntries[i].last_access<oldAccess)) {
|
|
||||||
if(cacheEntries[i].sector==CACHE_FREE) foundFree = true;
|
|
||||||
oldUsed = i;
|
|
||||||
oldAccess = cacheEntries[i].last_access;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if(foundFree==false && cacheEntries[oldUsed].dirty==true) {
|
|
||||||
if(!cache->disc->writeSectors(cacheEntries[oldUsed].sector,cacheEntries[oldUsed].count,cacheEntries[oldUsed].cache)) return NULL;
|
|
||||||
cacheEntries[oldUsed].dirty = false;
|
|
||||||
}
|
|
||||||
sector = (sector/sectorsPerPage)*sectorsPerPage; // align base sector to page size
|
|
||||||
sec_t next_page = sector + sectorsPerPage;
|
|
||||||
if(next_page > cache->endOfPartition) next_page = cache->endOfPartition;
|
|
||||||
|
|
||||||
if(!cache->disc->readSectors(sector,next_page-sector,cacheEntries[oldUsed].cache)) return NULL;
|
|
||||||
|
|
||||||
cacheEntries[oldUsed].sector = sector;
|
|
||||||
cacheEntries[oldUsed].count = next_page-sector;
|
|
||||||
cacheEntries[oldUsed].last_access = accessTime();
|
|
||||||
|
|
||||||
return &(cacheEntries[oldUsed]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static NTFS_CACHE_ENTRY* _NTFS_cache_findPage(NTFS_CACHE *cache, sec_t sector, sec_t count) {
|
static NTFS_CACHE_ENTRY* _NTFS_cache_getPage(NTFS_CACHE *cache, sec_t sector)
|
||||||
|
|
||||||
unsigned int i;
|
|
||||||
NTFS_CACHE_ENTRY* cacheEntries = cache->cacheEntries;
|
|
||||||
unsigned int numberOfPages = cache->numberOfPages;
|
|
||||||
NTFS_CACHE_ENTRY *entry = NULL;
|
|
||||||
sec_t lowest = UINT_MAX;
|
|
||||||
|
|
||||||
for(i=0;i<numberOfPages;i++) {
|
|
||||||
if (cacheEntries[i].sector != CACHE_FREE) {
|
|
||||||
bool intersect;
|
|
||||||
if (sector > cacheEntries[i].sector) {
|
|
||||||
intersect = sector - cacheEntries[i].sector < cacheEntries[i].count;
|
|
||||||
} else {
|
|
||||||
intersect = cacheEntries[i].sector - sector < count;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( intersect && (cacheEntries[i].sector < lowest)) {
|
|
||||||
lowest = cacheEntries[i].sector;
|
|
||||||
entry = &cacheEntries[i];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return entry;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool _NTFS_cache_readSectors(NTFS_CACHE *cache,sec_t sector,sec_t numSectors,void *buffer)
|
|
||||||
{
|
{
|
||||||
sec_t sec;
|
unsigned int i;
|
||||||
sec_t secs_to_read;
|
NTFS_CACHE_ENTRY* cacheEntries = cache->cacheEntries;
|
||||||
NTFS_CACHE_ENTRY *entry;
|
unsigned int numberOfPages = cache->numberOfPages;
|
||||||
uint8_t *dest = buffer;
|
unsigned int sectorsPerPage = cache->sectorsPerPage;
|
||||||
|
|
||||||
while(numSectors>0) {
|
bool foundFree = false;
|
||||||
entry = _NTFS_cache_getPage(cache,sector);
|
unsigned int oldUsed = 0;
|
||||||
if(entry==NULL) return false;
|
unsigned int oldAccess = UINT_MAX;
|
||||||
|
|
||||||
sec = sector - entry->sector;
|
for (i = 0; i < numberOfPages; i++)
|
||||||
secs_to_read = entry->count - sec;
|
{
|
||||||
if(secs_to_read>numSectors) secs_to_read = numSectors;
|
if (sector >= cacheEntries[i].sector && sector < (cacheEntries[i].sector + cacheEntries[i].count))
|
||||||
|
{
|
||||||
|
cacheEntries[i].last_access = accessTime();
|
||||||
|
return &(cacheEntries[i]);
|
||||||
|
}
|
||||||
|
|
||||||
memcpy(dest,entry->cache + (sec*cache->sectorSize),(secs_to_read*cache->sectorSize));
|
if (foundFree == false && (cacheEntries[i].sector == CACHE_FREE || cacheEntries[i].last_access < oldAccess))
|
||||||
|
{
|
||||||
|
if (cacheEntries[i].sector == CACHE_FREE) foundFree = true;
|
||||||
|
oldUsed = i;
|
||||||
|
oldAccess = cacheEntries[i].last_access;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
dest += (secs_to_read*cache->sectorSize);
|
if (foundFree == false && cacheEntries[oldUsed].dirty == true)
|
||||||
sector += secs_to_read;
|
{
|
||||||
numSectors -= secs_to_read;
|
if (!cache->disc->writeSectors(cacheEntries[oldUsed].sector, cacheEntries[oldUsed].count,
|
||||||
}
|
cacheEntries[oldUsed].cache)) return NULL;
|
||||||
|
cacheEntries[oldUsed].dirty = false;
|
||||||
|
}
|
||||||
|
sector = (sector / sectorsPerPage) * sectorsPerPage; // align base sector to page size
|
||||||
|
sec_t next_page = sector + sectorsPerPage;
|
||||||
|
if (next_page > cache->endOfPartition) next_page = cache->endOfPartition;
|
||||||
|
|
||||||
return true;
|
if (!cache->disc->readSectors(sector, next_page - sector, cacheEntries[oldUsed].cache)) return NULL;
|
||||||
|
|
||||||
|
cacheEntries[oldUsed].sector = sector;
|
||||||
|
cacheEntries[oldUsed].count = next_page - sector;
|
||||||
|
cacheEntries[oldUsed].last_access = accessTime();
|
||||||
|
|
||||||
|
return &(cacheEntries[oldUsed]);
|
||||||
|
}
|
||||||
|
|
||||||
|
static NTFS_CACHE_ENTRY* _NTFS_cache_findPage(NTFS_CACHE *cache, sec_t sector, sec_t count)
|
||||||
|
{
|
||||||
|
|
||||||
|
unsigned int i;
|
||||||
|
NTFS_CACHE_ENTRY* cacheEntries = cache->cacheEntries;
|
||||||
|
unsigned int numberOfPages = cache->numberOfPages;
|
||||||
|
NTFS_CACHE_ENTRY *entry = NULL;
|
||||||
|
sec_t lowest = UINT_MAX;
|
||||||
|
|
||||||
|
for (i = 0; i < numberOfPages; i++)
|
||||||
|
{
|
||||||
|
if (cacheEntries[i].sector != CACHE_FREE)
|
||||||
|
{
|
||||||
|
bool intersect;
|
||||||
|
if (sector > cacheEntries[i].sector)
|
||||||
|
{
|
||||||
|
intersect = sector - cacheEntries[i].sector < cacheEntries[i].count;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
intersect = cacheEntries[i].sector - sector < count;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (intersect && (cacheEntries[i].sector < lowest))
|
||||||
|
{
|
||||||
|
lowest = cacheEntries[i].sector;
|
||||||
|
entry = &cacheEntries[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool _NTFS_cache_readSectors(NTFS_CACHE *cache, sec_t sector, sec_t numSectors, void *buffer)
|
||||||
|
{
|
||||||
|
sec_t sec;
|
||||||
|
sec_t secs_to_read;
|
||||||
|
NTFS_CACHE_ENTRY *entry;
|
||||||
|
uint8_t *dest = buffer;
|
||||||
|
|
||||||
|
while (numSectors > 0)
|
||||||
|
{
|
||||||
|
entry = _NTFS_cache_getPage(cache, sector);
|
||||||
|
if (entry == NULL) return false;
|
||||||
|
|
||||||
|
sec = sector - entry->sector;
|
||||||
|
secs_to_read = entry->count - sec;
|
||||||
|
if (secs_to_read > numSectors) secs_to_read = numSectors;
|
||||||
|
|
||||||
|
memcpy(dest, entry->cache + (sec * cache->sectorSize), (secs_to_read * cache->sectorSize));
|
||||||
|
|
||||||
|
dest += (secs_to_read * cache->sectorSize);
|
||||||
|
sector += secs_to_read;
|
||||||
|
numSectors -= secs_to_read;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Reads some data from a cache page, determined by the sector number
|
Reads some data from a cache page, determined by the sector number
|
||||||
*/
|
*/
|
||||||
|
|
||||||
bool _NTFS_cache_readPartialSector (NTFS_CACHE* cache, void* buffer, sec_t sector, unsigned int offset, size_t size)
|
bool _NTFS_cache_readPartialSector(NTFS_CACHE* cache, void* buffer, sec_t sector, unsigned int offset, size_t size)
|
||||||
{
|
{
|
||||||
sec_t sec;
|
sec_t sec;
|
||||||
NTFS_CACHE_ENTRY *entry;
|
NTFS_CACHE_ENTRY *entry;
|
||||||
|
|
||||||
if (offset + size > cache->sectorSize) return false;
|
if (offset + size > cache->sectorSize) return false;
|
||||||
|
|
||||||
entry = _NTFS_cache_getPage(cache,sector);
|
entry = _NTFS_cache_getPage(cache, sector);
|
||||||
if(entry==NULL) return false;
|
if (entry == NULL) return false;
|
||||||
|
|
||||||
sec = sector - entry->sector;
|
sec = sector - entry->sector;
|
||||||
memcpy(buffer,entry->cache + ((sec*cache->sectorSize) + offset),size);
|
memcpy(buffer, entry->cache + ((sec * cache->sectorSize) + offset), size);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool _NTFS_cache_readLittleEndianValue (NTFS_CACHE* cache, uint32_t *value, sec_t sector, unsigned int offset, int num_bytes) {
|
bool _NTFS_cache_readLittleEndianValue(NTFS_CACHE* cache, uint32_t *value, sec_t sector, unsigned int offset,
|
||||||
uint8_t buf[4];
|
int num_bytes)
|
||||||
if (!_NTFS_cache_readPartialSector(cache, buf, sector, offset, num_bytes)) return false;
|
{
|
||||||
|
uint8_t buf[4];
|
||||||
|
if (!_NTFS_cache_readPartialSector(cache, buf, sector, offset, num_bytes)) return false;
|
||||||
|
|
||||||
switch(num_bytes) {
|
switch (num_bytes)
|
||||||
case 1: *value = buf[0]; break;
|
{
|
||||||
case 2: *value = u8array_to_u16(buf,0); break;
|
case 1:
|
||||||
case 4: *value = u8array_to_u32(buf,0); break;
|
*value = buf[0];
|
||||||
default: return false;
|
break;
|
||||||
}
|
case 2:
|
||||||
return true;
|
*value = u8array_to_u16(buf, 0);
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
*value = u8array_to_u32(buf, 0);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Writes some data to a cache page, making sure it is loaded into memory first.
|
Writes some data to a cache page, making sure it is loaded into memory first.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
bool _NTFS_cache_writePartialSector (NTFS_CACHE* cache, const void* buffer, sec_t sector, unsigned int offset, size_t size)
|
bool _NTFS_cache_writePartialSector(NTFS_CACHE* cache, const void* buffer, sec_t sector, unsigned int offset,
|
||||||
|
size_t size)
|
||||||
{
|
{
|
||||||
sec_t sec;
|
sec_t sec;
|
||||||
NTFS_CACHE_ENTRY *entry;
|
NTFS_CACHE_ENTRY *entry;
|
||||||
|
|
||||||
if (offset + size > cache->sectorSize) return false;
|
if (offset + size > cache->sectorSize) return false;
|
||||||
|
|
||||||
entry = _NTFS_cache_getPage(cache,sector);
|
entry = _NTFS_cache_getPage(cache, sector);
|
||||||
if(entry==NULL) return false;
|
if (entry == NULL) return false;
|
||||||
|
|
||||||
sec = sector - entry->sector;
|
sec = sector - entry->sector;
|
||||||
memcpy(entry->cache + ((sec*cache->sectorSize) + offset),buffer,size);
|
memcpy(entry->cache + ((sec * cache->sectorSize) + offset), buffer, size);
|
||||||
|
|
||||||
entry->dirty = true;
|
entry->dirty = true;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool _NTFS_cache_writeLittleEndianValue (NTFS_CACHE* cache, const uint32_t value, sec_t sector, unsigned int offset, int size) {
|
bool _NTFS_cache_writeLittleEndianValue(NTFS_CACHE* cache, const uint32_t value, sec_t sector, unsigned int offset,
|
||||||
uint8_t buf[4] = {0, 0, 0, 0};
|
int size)
|
||||||
|
{
|
||||||
|
uint8_t buf[4] = { 0, 0, 0, 0 };
|
||||||
|
|
||||||
switch(size) {
|
switch (size)
|
||||||
case 1: buf[0] = value; break;
|
{
|
||||||
case 2: u16_to_u8array(buf, 0, value); break;
|
case 1:
|
||||||
case 4: u32_to_u8array(buf, 0, value); break;
|
buf[0] = value;
|
||||||
default: return false;
|
break;
|
||||||
}
|
case 2:
|
||||||
|
u16_to_u8array(buf, 0, value);
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
u32_to_u8array(buf, 0, value);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
return _NTFS_cache_writePartialSector(cache, buf, sector, offset, size);
|
return _NTFS_cache_writePartialSector(cache, buf, sector, offset, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Writes some data to a cache page, zeroing out the page first
|
Writes some data to a cache page, zeroing out the page first
|
||||||
*/
|
*/
|
||||||
|
|
||||||
bool _NTFS_cache_eraseWritePartialSector (NTFS_CACHE* cache, const void* buffer, sec_t sector, unsigned int offset, size_t size)
|
bool _NTFS_cache_eraseWritePartialSector(NTFS_CACHE* cache, const void* buffer, sec_t sector, unsigned int offset,
|
||||||
|
size_t size)
|
||||||
{
|
{
|
||||||
sec_t sec;
|
sec_t sec;
|
||||||
NTFS_CACHE_ENTRY *entry;
|
NTFS_CACHE_ENTRY *entry;
|
||||||
|
|
||||||
if (offset + size > cache->sectorSize) return false;
|
if (offset + size > cache->sectorSize) return false;
|
||||||
|
|
||||||
entry = _NTFS_cache_getPage(cache,sector);
|
entry = _NTFS_cache_getPage(cache, sector);
|
||||||
if(entry==NULL) return false;
|
if (entry == NULL) return false;
|
||||||
|
|
||||||
sec = sector - entry->sector;
|
sec = sector - entry->sector;
|
||||||
memset(entry->cache + (sec*cache->sectorSize),0,cache->sectorSize);
|
memset(entry->cache + (sec * cache->sectorSize), 0, cache->sectorSize);
|
||||||
memcpy(entry->cache + ((sec*cache->sectorSize) + offset),buffer,size);
|
memcpy(entry->cache + ((sec * cache->sectorSize) + offset), buffer, size);
|
||||||
|
|
||||||
entry->dirty = true;
|
entry->dirty = true;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool _NTFS_cache_writeSectors (NTFS_CACHE* cache, sec_t sector, sec_t numSectors, const void* buffer)
|
bool _NTFS_cache_writeSectors(NTFS_CACHE* cache, sec_t sector, sec_t numSectors, const void* buffer)
|
||||||
{
|
{
|
||||||
sec_t sec;
|
sec_t sec;
|
||||||
sec_t secs_to_write;
|
sec_t secs_to_write;
|
||||||
NTFS_CACHE_ENTRY* entry;
|
NTFS_CACHE_ENTRY* entry;
|
||||||
const uint8_t *src = buffer;
|
const uint8_t *src = buffer;
|
||||||
|
|
||||||
while(numSectors>0)
|
while (numSectors > 0)
|
||||||
{
|
{
|
||||||
entry = _NTFS_cache_findPage(cache,sector,numSectors);
|
entry = _NTFS_cache_findPage(cache, sector, numSectors);
|
||||||
|
|
||||||
if(entry!=NULL) {
|
if (entry != NULL)
|
||||||
|
{
|
||||||
|
|
||||||
if ( entry->sector > sector) {
|
if (entry->sector > sector)
|
||||||
|
{
|
||||||
|
|
||||||
secs_to_write = entry->sector - sector;
|
secs_to_write = entry->sector - sector;
|
||||||
|
|
||||||
cache->disc->writeSectors(sector,secs_to_write,src);
|
cache->disc->writeSectors(sector, secs_to_write, src);
|
||||||
src += (secs_to_write*cache->sectorSize);
|
src += (secs_to_write * cache->sectorSize);
|
||||||
sector += secs_to_write;
|
sector += secs_to_write;
|
||||||
numSectors -= secs_to_write;
|
numSectors -= secs_to_write;
|
||||||
}
|
}
|
||||||
|
|
||||||
sec = sector - entry->sector;
|
sec = sector - entry->sector;
|
||||||
secs_to_write = entry->count - sec;
|
secs_to_write = entry->count - sec;
|
||||||
|
|
||||||
if(secs_to_write>numSectors) secs_to_write = numSectors;
|
if (secs_to_write > numSectors) secs_to_write = numSectors;
|
||||||
|
|
||||||
memcpy(entry->cache + (sec*cache->sectorSize),src,(secs_to_write*cache->sectorSize));
|
memcpy(entry->cache + (sec * cache->sectorSize), src, (secs_to_write * cache->sectorSize));
|
||||||
|
|
||||||
src += (secs_to_write*cache->sectorSize);
|
src += (secs_to_write * cache->sectorSize);
|
||||||
sector += secs_to_write;
|
sector += secs_to_write;
|
||||||
numSectors -= secs_to_write;
|
numSectors -= secs_to_write;
|
||||||
|
|
||||||
entry->dirty = true;
|
entry->dirty = true;
|
||||||
|
|
||||||
} else {
|
}
|
||||||
cache->disc->writeSectors(sector,numSectors,src);
|
else
|
||||||
numSectors=0;
|
{
|
||||||
}
|
cache->disc->writeSectors(sector, numSectors, src);
|
||||||
}
|
numSectors = 0;
|
||||||
return true;
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Flushes all dirty pages to disc, clearing the dirty flag.
|
Flushes all dirty pages to disc, clearing the dirty flag.
|
||||||
*/
|
*/
|
||||||
bool _NTFS_cache_flush (NTFS_CACHE* cache) {
|
bool _NTFS_cache_flush(NTFS_CACHE* cache)
|
||||||
unsigned int i;
|
{
|
||||||
if(cache==NULL) return true;
|
unsigned int i;
|
||||||
|
if (cache == NULL) return true;
|
||||||
|
|
||||||
for (i = 0; i < cache->numberOfPages; i++) {
|
for (i = 0; i < cache->numberOfPages; i++)
|
||||||
if (cache->cacheEntries[i].dirty) {
|
{
|
||||||
if (!cache->disc->writeSectors (cache->cacheEntries[i].sector, cache->cacheEntries[i].count, cache->cacheEntries[i].cache)) {
|
if (cache->cacheEntries[i].dirty)
|
||||||
return false;
|
{
|
||||||
}
|
if (!cache->disc->writeSectors(cache->cacheEntries[i].sector, cache->cacheEntries[i].count,
|
||||||
}
|
cache->cacheEntries[i].cache))
|
||||||
cache->cacheEntries[i].dirty = false;
|
{
|
||||||
}
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cache->cacheEntries[i].dirty = false;
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void _NTFS_cache_invalidate (NTFS_CACHE* cache) {
|
void _NTFS_cache_invalidate(NTFS_CACHE* cache)
|
||||||
unsigned int i;
|
{
|
||||||
if(cache==NULL)
|
unsigned int i;
|
||||||
return;
|
if (cache == NULL) return;
|
||||||
|
|
||||||
_NTFS_cache_flush(cache);
|
_NTFS_cache_flush(cache);
|
||||||
for (i = 0; i < cache->numberOfPages; i++) {
|
for (i = 0; i < cache->numberOfPages; i++)
|
||||||
cache->cacheEntries[i].sector = CACHE_FREE;
|
{
|
||||||
cache->cacheEntries[i].last_access = 0;
|
cache->cacheEntries[i].sector = CACHE_FREE;
|
||||||
cache->cacheEntries[i].count = 0;
|
cache->cacheEntries[i].last_access = 0;
|
||||||
cache->cacheEntries[i].dirty = false;
|
cache->cacheEntries[i].count = 0;
|
||||||
}
|
cache->cacheEntries[i].dirty = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -15,13 +15,13 @@
|
|||||||
Redistribution and use in source and binary forms, with or without modification,
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
are permitted provided that the following conditions are met:
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice,
|
1. Redistributions of source code must retain the above copyright notice,
|
||||||
this list of conditions and the following disclaimer.
|
this list of conditions and the following disclaimer.
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
this list of conditions and the following disclaimer in the documentation and/or
|
this list of conditions and the following disclaimer in the documentation and/or
|
||||||
other materials provided with the distribution.
|
other materials provided with the distribution.
|
||||||
3. The name of the author may not be used to endorse or promote products derived
|
3. The name of the author may not be used to endorse or promote products derived
|
||||||
from this software without specific prior written permission.
|
from this software without specific prior written permission.
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
||||||
@ -32,7 +32,7 @@
|
|||||||
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef _CACHE2_H
|
#ifndef _CACHE2_H
|
||||||
#define _CACHE2_H
|
#define _CACHE2_H
|
||||||
@ -46,90 +46,92 @@
|
|||||||
#include <ogc/disc_io.h>
|
#include <ogc/disc_io.h>
|
||||||
#include <gccore.h>
|
#include <gccore.h>
|
||||||
|
|
||||||
typedef struct {
|
typedef struct
|
||||||
sec_t sector;
|
{
|
||||||
unsigned int count;
|
sec_t sector;
|
||||||
u64 last_access;
|
unsigned int count;
|
||||||
bool dirty;
|
u64 last_access;
|
||||||
u8* cache;
|
bool dirty;
|
||||||
|
u8* cache;
|
||||||
} NTFS_CACHE_ENTRY;
|
} NTFS_CACHE_ENTRY;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct
|
||||||
const DISC_INTERFACE* disc;
|
{
|
||||||
sec_t endOfPartition;
|
const DISC_INTERFACE* disc;
|
||||||
unsigned int numberOfPages;
|
sec_t endOfPartition;
|
||||||
unsigned int sectorsPerPage;
|
unsigned int numberOfPages;
|
||||||
sec_t sectorSize;
|
unsigned int sectorsPerPage;
|
||||||
NTFS_CACHE_ENTRY* cacheEntries;
|
sec_t sectorSize;
|
||||||
|
NTFS_CACHE_ENTRY* cacheEntries;
|
||||||
} NTFS_CACHE;
|
} NTFS_CACHE;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Read data from a sector in the NTFS_CACHE
|
Read data from a sector in the NTFS_CACHE
|
||||||
If the sector is not in the NTFS_CACHE, it will be swapped in
|
If the sector is not in the NTFS_CACHE, it will be swapped in
|
||||||
offset is the position to start reading from
|
offset is the position to start reading from
|
||||||
size is the amount of data to read
|
size is the amount of data to read
|
||||||
Precondition: offset + size <= BYTES_PER_READ
|
Precondition: offset + size <= BYTES_PER_READ
|
||||||
*/
|
*/
|
||||||
//bool _NTFS_cache_readPartialSector (NTFS_CACHE* NTFS_CACHE, void* buffer, sec_t sector, unsigned int offset, size_t size);
|
//bool _NTFS_cache_readPartialSector (NTFS_CACHE* NTFS_CACHE, void* buffer, sec_t sector, unsigned int offset, size_t size);
|
||||||
|
|
||||||
//bool _NTFS_cache_readLittleEndianValue (NTFS_CACHE* NTFS_CACHE, uint32_t *value, sec_t sector, unsigned int offset, int num_bytes);
|
//bool _NTFS_cache_readLittleEndianValue (NTFS_CACHE* NTFS_CACHE, uint32_t *value, sec_t sector, unsigned int offset, int num_bytes);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write data to a sector in the NTFS_CACHE
|
Write data to a sector in the NTFS_CACHE
|
||||||
If the sector is not in the NTFS_CACHE, it will be swapped in.
|
If the sector is not in the NTFS_CACHE, it will be swapped in.
|
||||||
When the sector is swapped out, the data will be written to the disc
|
When the sector is swapped out, the data will be written to the disc
|
||||||
offset is the position to start writing to
|
offset is the position to start writing to
|
||||||
size is the amount of data to write
|
size is the amount of data to write
|
||||||
Precondition: offset + size <= BYTES_PER_READ
|
Precondition: offset + size <= BYTES_PER_READ
|
||||||
*/
|
*/
|
||||||
//bool _NTFS_cache_writePartialSector (NTFS_CACHE* NTFS_CACHE, const void* buffer, sec_t sector, unsigned int offset, size_t size);
|
//bool _NTFS_cache_writePartialSector (NTFS_CACHE* NTFS_CACHE, const void* buffer, sec_t sector, unsigned int offset, size_t size);
|
||||||
|
|
||||||
//bool _NTFS_cache_writeLittleEndianValue (NTFS_CACHE* NTFS_CACHE, const uint32_t value, sec_t sector, unsigned int offset, int num_bytes);
|
//bool _NTFS_cache_writeLittleEndianValue (NTFS_CACHE* NTFS_CACHE, const uint32_t value, sec_t sector, unsigned int offset, int num_bytes);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write data to a sector in the NTFS_CACHE, zeroing the sector first
|
Write data to a sector in the NTFS_CACHE, zeroing the sector first
|
||||||
If the sector is not in the NTFS_CACHE, it will be swapped in.
|
If the sector is not in the NTFS_CACHE, it will be swapped in.
|
||||||
When the sector is swapped out, the data will be written to the disc
|
When the sector is swapped out, the data will be written to the disc
|
||||||
offset is the position to start writing to
|
offset is the position to start writing to
|
||||||
size is the amount of data to write
|
size is the amount of data to write
|
||||||
Precondition: offset + size <= BYTES_PER_READ
|
Precondition: offset + size <= BYTES_PER_READ
|
||||||
*/
|
*/
|
||||||
//bool _NTFS_cache_eraseWritePartialSector (NTFS_CACHE* NTFS_CACHE, const void* buffer, sec_t sector, unsigned int offset, size_t size);
|
//bool _NTFS_cache_eraseWritePartialSector (NTFS_CACHE* NTFS_CACHE, const void* buffer, sec_t sector, unsigned int offset, size_t size);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Read several sectors from the NTFS_CACHE
|
Read several sectors from the NTFS_CACHE
|
||||||
*/
|
*/
|
||||||
bool _NTFS_cache_readSectors (NTFS_CACHE* NTFS_CACHE, sec_t sector, sec_t numSectors, void* buffer);
|
bool _NTFS_cache_readSectors(NTFS_CACHE* NTFS_CACHE, sec_t sector, sec_t numSectors, void* buffer);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Read a full sector from the NTFS_CACHE
|
Read a full sector from the NTFS_CACHE
|
||||||
*/
|
*/
|
||||||
//static inline bool _NTFS_cache_readSector (NTFS_CACHE* NTFS_CACHE, void* buffer, sec_t sector) {
|
//static inline bool _NTFS_cache_readSector (NTFS_CACHE* NTFS_CACHE, void* buffer, sec_t sector) {
|
||||||
// return _NTFS_cache_readPartialSector (NTFS_CACHE, buffer, sector, 0, BYTES_PER_READ);
|
// return _NTFS_cache_readPartialSector (NTFS_CACHE, buffer, sector, 0, BYTES_PER_READ);
|
||||||
//}
|
//}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write a full sector to the NTFS_CACHE
|
Write a full sector to the NTFS_CACHE
|
||||||
*/
|
*/
|
||||||
//static inline bool _NTFS_cache_writeSector (NTFS_CACHE* NTFS_CACHE, const void* buffer, sec_t sector) {
|
//static inline bool _NTFS_cache_writeSector (NTFS_CACHE* NTFS_CACHE, const void* buffer, sec_t sector) {
|
||||||
// return _NTFS_cache_writePartialSector (NTFS_CACHE, buffer, sector, 0, BYTES_PER_READ);
|
// return _NTFS_cache_writePartialSector (NTFS_CACHE, buffer, sector, 0, BYTES_PER_READ);
|
||||||
//}
|
//}
|
||||||
|
|
||||||
bool _NTFS_cache_writeSectors (NTFS_CACHE* NTFS_CACHE, sec_t sector, sec_t numSectors, const void* buffer);
|
bool _NTFS_cache_writeSectors(NTFS_CACHE* NTFS_CACHE, sec_t sector, sec_t numSectors, const void* buffer);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Write any dirty sectors back to disc and clear out the contents of the NTFS_CACHE
|
Write any dirty sectors back to disc and clear out the contents of the NTFS_CACHE
|
||||||
*/
|
*/
|
||||||
bool _NTFS_cache_flush (NTFS_CACHE* NTFS_CACHE);
|
bool _NTFS_cache_flush(NTFS_CACHE* NTFS_CACHE);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Clear out the contents of the NTFS_CACHE without writing any dirty sectors first
|
Clear out the contents of the NTFS_CACHE without writing any dirty sectors first
|
||||||
*/
|
*/
|
||||||
void _NTFS_cache_invalidate (NTFS_CACHE* NTFS_CACHE);
|
void _NTFS_cache_invalidate(NTFS_CACHE* NTFS_CACHE);
|
||||||
|
|
||||||
NTFS_CACHE* _NTFS_cache_constructor (unsigned int numberOfPages, unsigned int sectorsPerPage, const DISC_INTERFACE* discInterface, sec_t endOfPartition, sec_t sectorSize);
|
NTFS_CACHE* _NTFS_cache_constructor(unsigned int numberOfPages, unsigned int sectorsPerPage,
|
||||||
|
const DISC_INTERFACE* discInterface, sec_t endOfPartition, sec_t sectorSize);
|
||||||
|
|
||||||
void _NTFS_cache_destructor (NTFS_CACHE* NTFS_CACHE);
|
void _NTFS_cache_destructor(NTFS_CACHE* NTFS_CACHE);
|
||||||
|
|
||||||
#endif // _CACHE_H
|
#endif // _CACHE_H
|
||||||
|
|
||||||
|
@ -52,22 +52,21 @@
|
|||||||
*
|
*
|
||||||
* Returns:
|
* Returns:
|
||||||
*/
|
*/
|
||||||
static int ntfs_collate_binary(ntfs_volume *vol __attribute__((unused)),
|
static int ntfs_collate_binary(ntfs_volume *vol __attribute__((unused)), const void *data1, const int data1_len, const void *data2,
|
||||||
const void *data1, const int data1_len,
|
const int data2_len)
|
||||||
const void *data2, const int data2_len)
|
|
||||||
{
|
{
|
||||||
int rc;
|
int rc;
|
||||||
|
|
||||||
ntfs_log_trace("Entering.\n");
|
ntfs_log_trace("Entering.\n");
|
||||||
rc = memcmp(data1, data2, min(data1_len, data2_len));
|
rc = memcmp(data1, data2, min(data1_len, data2_len));
|
||||||
if (!rc && (data1_len != data2_len)) {
|
if (!rc && (data1_len != data2_len))
|
||||||
if (data1_len < data2_len)
|
{
|
||||||
rc = -1;
|
if (data1_len < data2_len)
|
||||||
else
|
rc = -1;
|
||||||
rc = 1;
|
else rc = 1;
|
||||||
}
|
}
|
||||||
ntfs_log_trace("Done, returning %i.\n", rc);
|
ntfs_log_trace("Done, returning %i.\n", rc);
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -82,30 +81,30 @@ static int ntfs_collate_binary(ntfs_volume *vol __attribute__((unused)),
|
|||||||
*
|
*
|
||||||
* Returns:
|
* Returns:
|
||||||
*/
|
*/
|
||||||
static int ntfs_collate_ntofs_ulong(ntfs_volume *vol __attribute__((unused)),
|
static int ntfs_collate_ntofs_ulong(ntfs_volume *vol __attribute__((unused)), const void *data1, const int data1_len, const void *data2,
|
||||||
const void *data1, const int data1_len,
|
const int data2_len)
|
||||||
const void *data2, const int data2_len)
|
|
||||||
{
|
{
|
||||||
int rc;
|
int rc;
|
||||||
u32 d1, d2;
|
u32 d1, d2;
|
||||||
|
|
||||||
ntfs_log_trace("Entering.\n");
|
ntfs_log_trace("Entering.\n");
|
||||||
if (data1_len != data2_len || data1_len != 4) {
|
if (data1_len != data2_len || data1_len != 4)
|
||||||
ntfs_log_error("data1_len or/and data2_len not equal to 4.\n");
|
{
|
||||||
return NTFS_COLLATION_ERROR;
|
ntfs_log_error("data1_len or/and data2_len not equal to 4.\n");
|
||||||
}
|
return NTFS_COLLATION_ERROR;
|
||||||
d1 = le32_to_cpup(data1);
|
}
|
||||||
d2 = le32_to_cpup(data2);
|
d1 = le32_to_cpup(data1);
|
||||||
if (d1 < d2)
|
d2 = le32_to_cpup(data2);
|
||||||
rc = -1;
|
if (d1 < d2)
|
||||||
else {
|
rc = -1;
|
||||||
if (d1 == d2)
|
else
|
||||||
rc = 0;
|
{
|
||||||
else
|
if (d1 == d2)
|
||||||
rc = 1;
|
rc = 0;
|
||||||
}
|
else rc = 1;
|
||||||
ntfs_log_trace("Done, returning %i.\n", rc);
|
}
|
||||||
return rc;
|
ntfs_log_trace("Done, returning %i.\n", rc);
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -114,39 +113,40 @@ static int ntfs_collate_ntofs_ulong(ntfs_volume *vol __attribute__((unused)),
|
|||||||
* Returns: -1, 0 or 1 depending of how the arrays compare
|
* Returns: -1, 0 or 1 depending of how the arrays compare
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static int ntfs_collate_ntofs_ulongs(ntfs_volume *vol __attribute__((unused)),
|
static int ntfs_collate_ntofs_ulongs(ntfs_volume *vol __attribute__((unused)), const void *data1, const int data1_len, const void *data2,
|
||||||
const void *data1, const int data1_len,
|
const int data2_len)
|
||||||
const void *data2, const int data2_len)
|
|
||||||
{
|
{
|
||||||
int rc;
|
int rc;
|
||||||
int len;
|
int len;
|
||||||
const le32 *p1, *p2;
|
const le32 *p1, *p2;
|
||||||
u32 d1, d2;
|
u32 d1, d2;
|
||||||
|
|
||||||
ntfs_log_trace("Entering.\n");
|
ntfs_log_trace("Entering.\n");
|
||||||
if ((data1_len != data2_len) || (data1_len <= 0) || (data1_len & 3)) {
|
if ((data1_len != data2_len) || (data1_len <= 0) || (data1_len & 3))
|
||||||
ntfs_log_error("data1_len or data2_len not valid\n");
|
{
|
||||||
return NTFS_COLLATION_ERROR;
|
ntfs_log_error("data1_len or data2_len not valid\n");
|
||||||
}
|
return NTFS_COLLATION_ERROR;
|
||||||
p1 = (const le32*)data1;
|
}
|
||||||
p2 = (const le32*)data2;
|
p1 = (const le32*) data1;
|
||||||
len = data1_len;
|
p2 = (const le32*) data2;
|
||||||
do {
|
len = data1_len;
|
||||||
d1 = le32_to_cpup(p1);
|
do
|
||||||
p1++;
|
{
|
||||||
d2 = le32_to_cpup(p2);
|
d1 = le32_to_cpup(p1);
|
||||||
p2++;
|
p1++;
|
||||||
} while ((d1 == d2) && ((len -= 4) > 0));
|
d2 = le32_to_cpup(p2);
|
||||||
if (d1 < d2)
|
p2++;
|
||||||
rc = -1;
|
} while ((d1 == d2) && ((len -= 4) > 0));
|
||||||
else {
|
if (d1 < d2)
|
||||||
if (d1 == d2)
|
rc = -1;
|
||||||
rc = 0;
|
else
|
||||||
else
|
{
|
||||||
rc = 1;
|
if (d1 == d2)
|
||||||
}
|
rc = 0;
|
||||||
ntfs_log_trace("Done, returning %i.\n", rc);
|
else rc = 1;
|
||||||
return rc;
|
}
|
||||||
|
ntfs_log_trace("Done, returning %i.\n", rc);
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -162,45 +162,47 @@ static int ntfs_collate_ntofs_ulongs(ntfs_volume *vol __attribute__((unused)),
|
|||||||
*
|
*
|
||||||
* Returns: -1, 0 or 1 depending of how the keys compare
|
* Returns: -1, 0 or 1 depending of how the keys compare
|
||||||
*/
|
*/
|
||||||
static int ntfs_collate_ntofs_security_hash(ntfs_volume *vol __attribute__((unused)),
|
static int ntfs_collate_ntofs_security_hash(ntfs_volume *vol __attribute__((unused)), const void *data1, const int data1_len,
|
||||||
const void *data1, const int data1_len,
|
const void *data2, const int data2_len)
|
||||||
const void *data2, const int data2_len)
|
|
||||||
{
|
{
|
||||||
int rc;
|
int rc;
|
||||||
u32 d1, d2;
|
u32 d1, d2;
|
||||||
const le32 *p1, *p2;
|
const le32 *p1, *p2;
|
||||||
|
|
||||||
ntfs_log_trace("Entering.\n");
|
ntfs_log_trace("Entering.\n");
|
||||||
if (data1_len != data2_len || data1_len != 8) {
|
if (data1_len != data2_len || data1_len != 8)
|
||||||
ntfs_log_error("data1_len or/and data2_len not equal to 8.\n");
|
{
|
||||||
return NTFS_COLLATION_ERROR;
|
ntfs_log_error("data1_len or/and data2_len not equal to 8.\n");
|
||||||
}
|
return NTFS_COLLATION_ERROR;
|
||||||
p1 = (const le32*)data1;
|
}
|
||||||
p2 = (const le32*)data2;
|
p1 = (const le32*) data1;
|
||||||
d1 = le32_to_cpup(p1);
|
p2 = (const le32*) data2;
|
||||||
d2 = le32_to_cpup(p2);
|
d1 = le32_to_cpup(p1);
|
||||||
if (d1 < d2)
|
d2 = le32_to_cpup(p2);
|
||||||
rc = -1;
|
if (d1 < d2)
|
||||||
else {
|
rc = -1;
|
||||||
if (d1 > d2)
|
else
|
||||||
rc = 1;
|
{
|
||||||
else {
|
if (d1 > d2)
|
||||||
p1++;
|
rc = 1;
|
||||||
p2++;
|
else
|
||||||
d1 = le32_to_cpup(p1);
|
{
|
||||||
d2 = le32_to_cpup(p2);
|
p1++;
|
||||||
if (d1 < d2)
|
p2++;
|
||||||
rc = -1;
|
d1 = le32_to_cpup(p1);
|
||||||
else {
|
d2 = le32_to_cpup(p2);
|
||||||
if (d1 > d2)
|
if (d1 < d2)
|
||||||
rc = 1;
|
rc = -1;
|
||||||
else
|
else
|
||||||
rc = 0;
|
{
|
||||||
}
|
if (d1 > d2)
|
||||||
}
|
rc = 1;
|
||||||
}
|
else rc = 0;
|
||||||
ntfs_log_trace("Done, returning %i.\n", rc);
|
}
|
||||||
return rc;
|
}
|
||||||
|
}
|
||||||
|
ntfs_log_trace("Done, returning %i.\n", rc);
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -215,25 +217,21 @@ static int ntfs_collate_ntofs_security_hash(ntfs_volume *vol __attribute__((unus
|
|||||||
*
|
*
|
||||||
* Returns:
|
* Returns:
|
||||||
*/
|
*/
|
||||||
static int ntfs_collate_file_name(ntfs_volume *vol,
|
static int ntfs_collate_file_name(ntfs_volume *vol, const void *data1, const int data1_len __attribute__((unused)), const void *data2,
|
||||||
const void *data1, const int data1_len __attribute__((unused)),
|
const int data2_len __attribute__((unused)))
|
||||||
const void *data2, const int data2_len __attribute__((unused)))
|
|
||||||
{
|
{
|
||||||
const FILE_NAME_ATTR *file_name_attr1;
|
const FILE_NAME_ATTR *file_name_attr1;
|
||||||
const FILE_NAME_ATTR *file_name_attr2;
|
const FILE_NAME_ATTR *file_name_attr2;
|
||||||
int rc;
|
int rc;
|
||||||
|
|
||||||
ntfs_log_trace("Entering.\n");
|
ntfs_log_trace("Entering.\n");
|
||||||
file_name_attr1 = (const FILE_NAME_ATTR*)data1;
|
file_name_attr1 = (const FILE_NAME_ATTR*) data1;
|
||||||
file_name_attr2 = (const FILE_NAME_ATTR*)data2;
|
file_name_attr2 = (const FILE_NAME_ATTR*) data2;
|
||||||
rc = ntfs_names_full_collate(
|
rc = ntfs_names_full_collate((ntfschar*) &file_name_attr1->file_name, file_name_attr1->file_name_length,
|
||||||
(ntfschar*)&file_name_attr1->file_name,
|
(ntfschar*) &file_name_attr2->file_name, file_name_attr2->file_name_length, CASE_SENSITIVE, vol->upcase,
|
||||||
file_name_attr1->file_name_length,
|
vol->upcase_len);
|
||||||
(ntfschar*)&file_name_attr2->file_name,
|
ntfs_log_trace("Done, returning %i.\n", rc);
|
||||||
file_name_attr2->file_name_length,
|
return rc;
|
||||||
CASE_SENSITIVE, vol->upcase, vol->upcase_len);
|
|
||||||
ntfs_log_trace("Done, returning %i.\n", rc);
|
|
||||||
return rc;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -244,28 +242,29 @@ static int ntfs_collate_file_name(ntfs_volume *vol,
|
|||||||
|
|
||||||
COLLATE ntfs_get_collate_function(COLLATION_RULES cr)
|
COLLATE ntfs_get_collate_function(COLLATION_RULES cr)
|
||||||
{
|
{
|
||||||
COLLATE collate;
|
COLLATE collate;
|
||||||
|
|
||||||
switch (cr) {
|
switch (cr)
|
||||||
case COLLATION_BINARY :
|
{
|
||||||
collate = ntfs_collate_binary;
|
case COLLATION_BINARY:
|
||||||
break;
|
collate = ntfs_collate_binary;
|
||||||
case COLLATION_FILE_NAME :
|
break;
|
||||||
collate = ntfs_collate_file_name;
|
case COLLATION_FILE_NAME:
|
||||||
break;
|
collate = ntfs_collate_file_name;
|
||||||
case COLLATION_NTOFS_SECURITY_HASH :
|
break;
|
||||||
collate = ntfs_collate_ntofs_security_hash;
|
case COLLATION_NTOFS_SECURITY_HASH:
|
||||||
break;
|
collate = ntfs_collate_ntofs_security_hash;
|
||||||
case COLLATION_NTOFS_ULONG :
|
break;
|
||||||
collate = ntfs_collate_ntofs_ulong;
|
case COLLATION_NTOFS_ULONG:
|
||||||
break;
|
collate = ntfs_collate_ntofs_ulong;
|
||||||
case COLLATION_NTOFS_ULONGS :
|
break;
|
||||||
collate = ntfs_collate_ntofs_ulongs;
|
case COLLATION_NTOFS_ULONGS:
|
||||||
break;
|
collate = ntfs_collate_ntofs_ulongs;
|
||||||
default :
|
break;
|
||||||
errno = EOPNOTSUPP;
|
default:
|
||||||
collate = (COLLATE)NULL;
|
errno = EOPNOTSUPP;
|
||||||
break;
|
collate = (COLLATE) NULL;
|
||||||
}
|
break;
|
||||||
return (collate);
|
}
|
||||||
|
return (collate);
|
||||||
}
|
}
|
||||||
|
@ -37,31 +37,35 @@
|
|||||||
*/
|
*/
|
||||||
int ffs(int x)
|
int ffs(int x)
|
||||||
{
|
{
|
||||||
int r = 1;
|
int r = 1;
|
||||||
|
|
||||||
if (!x)
|
if (!x) return 0;
|
||||||
return 0;
|
if (!(x & 0xffff))
|
||||||
if (!(x & 0xffff)) {
|
{
|
||||||
x >>= 16;
|
x >>= 16;
|
||||||
r += 16;
|
r += 16;
|
||||||
}
|
}
|
||||||
if (!(x & 0xff)) {
|
if (!(x & 0xff))
|
||||||
x >>= 8;
|
{
|
||||||
r += 8;
|
x >>= 8;
|
||||||
}
|
r += 8;
|
||||||
if (!(x & 0xf)) {
|
}
|
||||||
x >>= 4;
|
if (!(x & 0xf))
|
||||||
r += 4;
|
{
|
||||||
}
|
x >>= 4;
|
||||||
if (!(x & 3)) {
|
r += 4;
|
||||||
x >>= 2;
|
}
|
||||||
r += 2;
|
if (!(x & 3))
|
||||||
}
|
{
|
||||||
if (!(x & 1)) {
|
x >>= 2;
|
||||||
x >>= 1;
|
r += 2;
|
||||||
r += 1;
|
}
|
||||||
}
|
if (!(x & 1))
|
||||||
return r;
|
{
|
||||||
|
x >>= 1;
|
||||||
|
r += 1;
|
||||||
|
}
|
||||||
|
return r;
|
||||||
}
|
}
|
||||||
#endif /* HAVE_FFS */
|
#endif /* HAVE_FFS */
|
||||||
|
|
||||||
@ -120,32 +124,32 @@ static const char rcsid[] = "$Id: compat.c,v 1.1.1.1.2.1 2008-08-16 15:17:44 jpa
|
|||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
int daemon(int nochdir, int noclose) {
|
int daemon(int nochdir, int noclose)
|
||||||
int fd;
|
{
|
||||||
|
int fd;
|
||||||
|
|
||||||
switch (fork()) {
|
switch (fork())
|
||||||
case -1:
|
{
|
||||||
return (-1);
|
case -1:
|
||||||
case 0:
|
return (-1);
|
||||||
break;
|
case 0:
|
||||||
default:
|
break;
|
||||||
_exit(0);
|
default:
|
||||||
}
|
_exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
if (setsid() == -1)
|
if (setsid() == -1) return (-1);
|
||||||
return (-1);
|
|
||||||
|
|
||||||
if (!nochdir)
|
if (!nochdir) (void) chdir("/");
|
||||||
(void)chdir("/");
|
|
||||||
|
|
||||||
if (!noclose && (fd = open("/dev/null", O_RDWR, 0)) != -1) {
|
if (!noclose && (fd = open("/dev/null", O_RDWR, 0)) != -1)
|
||||||
(void)dup2(fd, 0);
|
{
|
||||||
(void)dup2(fd, 1);
|
(void) dup2(fd, 0);
|
||||||
(void)dup2(fd, 2);
|
(void) dup2(fd, 1);
|
||||||
if (fd > 2)
|
(void) dup2(fd, 2);
|
||||||
(void)close (fd);
|
if (fd > 2) (void) close(fd);
|
||||||
}
|
}
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
* End: src/lib/libresolv2/common/bsd/daemon.c
|
* End: src/lib/libresolv2/common/bsd/daemon.c
|
||||||
@ -218,29 +222,31 @@ static const char rcsid[] = "$Id: compat.c,v 1.1.1.1.2.1 2008-08-16 15:17:44 jpa
|
|||||||
*
|
*
|
||||||
* If *stringp is NULL, strsep returns NULL.
|
* If *stringp is NULL, strsep returns NULL.
|
||||||
*/
|
*/
|
||||||
char *strsep(char **stringp, const char *delim) {
|
char *strsep(char **stringp, const char *delim)
|
||||||
char *s;
|
{
|
||||||
const char *spanp;
|
char *s;
|
||||||
int c, sc;
|
const char *spanp;
|
||||||
char *tok;
|
int c, sc;
|
||||||
|
char *tok;
|
||||||
|
|
||||||
if ((s = *stringp) == NULL)
|
if ((s = *stringp) == NULL) return (NULL);
|
||||||
return (NULL);
|
for (tok = s;;)
|
||||||
for (tok = s;;) {
|
{
|
||||||
c = *s++;
|
c = *s++;
|
||||||
spanp = delim;
|
spanp = delim;
|
||||||
do {
|
do
|
||||||
if ((sc = *spanp++) == c) {
|
{
|
||||||
if (c == 0)
|
if ((sc = *spanp++) == c)
|
||||||
s = NULL;
|
{
|
||||||
else
|
if (c == 0)
|
||||||
s[-1] = 0;
|
s = NULL;
|
||||||
*stringp = s;
|
else s[-1] = 0;
|
||||||
return (tok);
|
*stringp = s;
|
||||||
}
|
return (tok);
|
||||||
} while (sc != 0);
|
}
|
||||||
}
|
} while (sc != 0);
|
||||||
/* NOTREACHED */
|
}
|
||||||
|
/* NOTREACHED */
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -26,16 +26,12 @@
|
|||||||
#include "types.h"
|
#include "types.h"
|
||||||
#include "attrib.h"
|
#include "attrib.h"
|
||||||
|
|
||||||
extern s64 ntfs_compressed_attr_pread(ntfs_attr *na, s64 pos, s64 count,
|
extern s64 ntfs_compressed_attr_pread(ntfs_attr *na, s64 pos, s64 count, void *b);
|
||||||
void *b);
|
|
||||||
|
|
||||||
extern s64 ntfs_compressed_pwrite(ntfs_attr *na, runlist_element *brl, s64 wpos,
|
extern s64 ntfs_compressed_pwrite(ntfs_attr *na, runlist_element *brl, s64 wpos, s64 offs, s64 to_write, s64 rounded,
|
||||||
s64 offs, s64 to_write, s64 rounded,
|
const void *b, int compressed_part, VCN *update_from);
|
||||||
const void *b, int compressed_part,
|
|
||||||
VCN *update_from);
|
|
||||||
|
|
||||||
extern int ntfs_compressed_close(ntfs_attr *na, runlist_element *brl,
|
extern int ntfs_compressed_close(ntfs_attr *na, runlist_element *brl, s64 offs, VCN *update_from);
|
||||||
s64 offs, VCN *update_from);
|
|
||||||
|
|
||||||
#endif /* defined _NTFS_COMPRESS_H */
|
#endif /* defined _NTFS_COMPRESS_H */
|
||||||
|
|
||||||
|
@ -133,7 +133,7 @@
|
|||||||
#undef HAVE_SETXATTR
|
#undef HAVE_SETXATTR
|
||||||
|
|
||||||
/* Define to 1 if `stat' has the bug that it succeeds when given the
|
/* Define to 1 if `stat' has the bug that it succeeds when given the
|
||||||
zero-length file name argument. */
|
zero-length file name argument. */
|
||||||
#define HAVE_STAT_EMPTY_STRING_BUG 1
|
#define HAVE_STAT_EMPTY_STRING_BUG 1
|
||||||
|
|
||||||
/* Define to 1 if you have the <stdarg.h> header file. */
|
/* Define to 1 if you have the <stdarg.h> header file. */
|
||||||
@ -203,7 +203,7 @@
|
|||||||
#undef HAVE_STRUCT_STAT_ST_RDEV
|
#undef HAVE_STRUCT_STAT_ST_RDEV
|
||||||
|
|
||||||
/* Define to 1 if your `struct stat' has `st_blocks'. Deprecated, use
|
/* Define to 1 if your `struct stat' has `st_blocks'. Deprecated, use
|
||||||
`HAVE_STRUCT_STAT_ST_BLOCKS' instead. */
|
`HAVE_STRUCT_STAT_ST_BLOCKS' instead. */
|
||||||
#undef HAVE_ST_BLOCKS
|
#undef HAVE_ST_BLOCKS
|
||||||
|
|
||||||
/* Define to 1 if you have the `sysconf' function. */
|
/* Define to 1 if you have the `sysconf' function. */
|
||||||
@ -279,7 +279,7 @@
|
|||||||
#undef IGNORE_MTAB
|
#undef IGNORE_MTAB
|
||||||
|
|
||||||
/* Define to 1 if `lstat' dereferences a symlink specified with a trailing
|
/* Define to 1 if `lstat' dereferences a symlink specified with a trailing
|
||||||
slash. */
|
slash. */
|
||||||
#undef LSTAT_FOLLOWS_SLASHED_SYMLINK
|
#undef LSTAT_FOLLOWS_SLASHED_SYMLINK
|
||||||
|
|
||||||
/* Define to 1 if your C compiler doesn't accept -c and -o together. */
|
/* Define to 1 if your C compiler doesn't accept -c and -o together. */
|
||||||
@ -340,11 +340,11 @@
|
|||||||
#undef WINDOWS
|
#undef WINDOWS
|
||||||
|
|
||||||
/* Define to 1 if your processor stores words with the most significant byte
|
/* Define to 1 if your processor stores words with the most significant byte
|
||||||
first (like Motorola and SPARC, unlike Intel and VAX). */
|
first (like Motorola and SPARC, unlike Intel and VAX). */
|
||||||
#define WORDS_BIGENDIAN 1
|
#define WORDS_BIGENDIAN 1
|
||||||
|
|
||||||
/* Define to 1 if your processor stores words with the least significant byte
|
/* Define to 1 if your processor stores words with the least significant byte
|
||||||
first (like Intel and VAX, unlike Motorola and SPARC). */
|
first (like Intel and VAX, unlike Motorola and SPARC). */
|
||||||
#undef WORDS_LITTLEENDIAN
|
#undef WORDS_LITTLEENDIAN
|
||||||
|
|
||||||
/* Number of bits in a file offset, on hosts where this is settable. */
|
/* Number of bits in a file offset, on hosts where this is settable. */
|
||||||
@ -362,7 +362,7 @@
|
|||||||
#undef _REENTRANT
|
#undef _REENTRANT
|
||||||
|
|
||||||
/* Define to `__inline__' or `__inline' if that's what the C compiler
|
/* Define to `__inline__' or `__inline' if that's what the C compiler
|
||||||
calls it, or to nothing if 'inline' is not supported under any name. */
|
calls it, or to nothing if 'inline' is not supported under any name. */
|
||||||
#ifndef __cplusplus
|
#ifndef __cplusplus
|
||||||
#define inline __inline__
|
#define inline __inline__
|
||||||
#endif
|
#endif
|
||||||
|
@ -44,35 +44,40 @@
|
|||||||
*/
|
*/
|
||||||
void ntfs_debug_runlist_dump(const runlist_element *rl)
|
void ntfs_debug_runlist_dump(const runlist_element *rl)
|
||||||
{
|
{
|
||||||
int i = 0;
|
int i = 0;
|
||||||
const char *lcn_str[5] = { "LCN_HOLE ", "LCN_RL_NOT_MAPPED",
|
const char *lcn_str[5] =
|
||||||
"LCN_ENOENT ", "LCN_EINVAL ",
|
{ "LCN_HOLE ", "LCN_RL_NOT_MAPPED",
|
||||||
"LCN_unknown " };
|
"LCN_ENOENT ", "LCN_EINVAL ",
|
||||||
|
"LCN_unknown "};
|
||||||
|
|
||||||
ntfs_log_debug("NTFS-fs DEBUG: Dumping runlist (values in hex):\n");
|
ntfs_log_debug("NTFS-fs DEBUG: Dumping runlist (values in hex):\n");
|
||||||
if (!rl) {
|
if (!rl)
|
||||||
ntfs_log_debug("Run list not present.\n");
|
{
|
||||||
return;
|
ntfs_log_debug("Run list not present.\n");
|
||||||
}
|
return;
|
||||||
ntfs_log_debug("VCN LCN Run length\n");
|
}
|
||||||
do {
|
ntfs_log_debug("VCN LCN Run length\n");
|
||||||
LCN lcn = (rl + i)->lcn;
|
do
|
||||||
|
{
|
||||||
|
LCN lcn = (rl + i)->lcn;
|
||||||
|
|
||||||
if (lcn < (LCN)0) {
|
if (lcn < (LCN)0)
|
||||||
int idx = -lcn - 1;
|
{
|
||||||
|
int idx = -lcn - 1;
|
||||||
|
|
||||||
if (idx > -LCN_EINVAL - 1)
|
if (idx > -LCN_EINVAL - 1)
|
||||||
idx = 4;
|
idx = 4;
|
||||||
ntfs_log_debug("%-16lld %s %-16lld%s\n",
|
ntfs_log_debug("%-16lld %s %-16lld%s\n",
|
||||||
(long long)rl[i].vcn, lcn_str[idx],
|
(long long)rl[i].vcn, lcn_str[idx],
|
||||||
(long long)rl[i].length,
|
(long long)rl[i].length,
|
||||||
rl[i].length ? "" : " (runlist end)");
|
rl[i].length ? "" : " (runlist end)");
|
||||||
} else
|
}
|
||||||
ntfs_log_debug("%-16lld %-16lld %-16lld%s\n",
|
else
|
||||||
(long long)rl[i].vcn, (long long)rl[i].lcn,
|
ntfs_log_debug("%-16lld %-16lld %-16lld%s\n",
|
||||||
(long long)rl[i].length,
|
(long long)rl[i].vcn, (long long)rl[i].lcn,
|
||||||
rl[i].length ? "" : " (runlist end)");
|
(long long)rl[i].length,
|
||||||
} while (rl[i++].length);
|
rl[i].length ? "" : " (runlist end)");
|
||||||
|
}while (rl[i++].length);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -33,7 +33,9 @@ struct _runlist_element;
|
|||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
extern void ntfs_debug_runlist_dump(const struct _runlist_element *rl);
|
extern void ntfs_debug_runlist_dump(const struct _runlist_element *rl);
|
||||||
#else
|
#else
|
||||||
static __inline__ void ntfs_debug_runlist_dump(const struct _runlist_element *rl __attribute__((unused))) {}
|
static __inline__ void ntfs_debug_runlist_dump(const struct _runlist_element *rl __attribute__((unused)))
|
||||||
|
{
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define NTFS_BUG(msg) \
|
#define NTFS_BUG(msg) \
|
||||||
|
@ -103,29 +103,32 @@
|
|||||||
* On success return a pointer to the allocated ntfs device structure and on
|
* On success return a pointer to the allocated ntfs device structure and on
|
||||||
* error return NULL with errno set to the error code returned by ntfs_malloc().
|
* error return NULL with errno set to the error code returned by ntfs_malloc().
|
||||||
*/
|
*/
|
||||||
struct ntfs_device *ntfs_device_alloc(const char *name, const long state,
|
struct ntfs_device *ntfs_device_alloc(const char *name, const long state, struct ntfs_device_operations *dops,
|
||||||
struct ntfs_device_operations *dops, void *priv_data)
|
void *priv_data)
|
||||||
{
|
{
|
||||||
struct ntfs_device *dev;
|
struct ntfs_device *dev;
|
||||||
|
|
||||||
if (!name) {
|
if (!name)
|
||||||
errno = EINVAL;
|
{
|
||||||
return NULL;
|
errno = EINVAL;
|
||||||
}
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
dev = ntfs_malloc(sizeof(struct ntfs_device));
|
dev = ntfs_malloc(sizeof(struct ntfs_device));
|
||||||
if (dev) {
|
if (dev)
|
||||||
if (!(dev->d_name = strdup(name))) {
|
{
|
||||||
int eo = errno;
|
if (!(dev->d_name = strdup(name)))
|
||||||
free(dev);
|
{
|
||||||
errno = eo;
|
int eo = errno;
|
||||||
return NULL;
|
free(dev);
|
||||||
}
|
errno = eo;
|
||||||
dev->d_ops = dops;
|
return NULL;
|
||||||
dev->d_state = state;
|
}
|
||||||
dev->d_private = priv_data;
|
dev->d_ops = dops;
|
||||||
}
|
dev->d_state = state;
|
||||||
return dev;
|
dev->d_private = priv_data;
|
||||||
|
}
|
||||||
|
return dev;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -141,17 +144,19 @@ struct ntfs_device *ntfs_device_alloc(const char *name, const long state,
|
|||||||
*/
|
*/
|
||||||
int ntfs_device_free(struct ntfs_device *dev)
|
int ntfs_device_free(struct ntfs_device *dev)
|
||||||
{
|
{
|
||||||
if (!dev) {
|
if (!dev)
|
||||||
errno = EINVAL;
|
{
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
if (NDevOpen(dev)) {
|
}
|
||||||
errno = EBUSY;
|
if (NDevOpen(dev))
|
||||||
return -1;
|
{
|
||||||
}
|
errno = EBUSY;
|
||||||
free(dev->d_name);
|
return -1;
|
||||||
free(dev);
|
}
|
||||||
return 0;
|
free(dev->d_name);
|
||||||
|
free(dev);
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -175,33 +180,32 @@ int ntfs_device_free(struct ntfs_device *dev)
|
|||||||
*/
|
*/
|
||||||
s64 ntfs_pread(struct ntfs_device *dev, const s64 pos, s64 count, void *b)
|
s64 ntfs_pread(struct ntfs_device *dev, const s64 pos, s64 count, void *b)
|
||||||
{
|
{
|
||||||
s64 br, total;
|
s64 br, total;
|
||||||
struct ntfs_device_operations *dops;
|
struct ntfs_device_operations *dops;
|
||||||
|
|
||||||
ntfs_log_trace("pos %lld, count %lld\n",(long long)pos,(long long)count);
|
ntfs_log_trace("pos %lld, count %lld\n",(long long)pos,(long long)count);
|
||||||
|
|
||||||
if (!b || count < 0 || pos < 0) {
|
|
||||||
errno = EINVAL;
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
if (!count)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
dops = dev->d_ops;
|
|
||||||
|
|
||||||
for (total = 0; count; count -= br, total += br) {
|
if (!b || count < 0 || pos < 0)
|
||||||
br = dops->pread(dev, (char*)b + total, count, pos + total);
|
{
|
||||||
/* If everything ok, continue. */
|
errno = EINVAL;
|
||||||
if (br > 0)
|
return -1;
|
||||||
continue;
|
}
|
||||||
/* If EOF or error return number of bytes read. */
|
if (!count) return 0;
|
||||||
if (!br || total)
|
|
||||||
return total;
|
dops = dev->d_ops;
|
||||||
/* Nothing read and error, return error status. */
|
|
||||||
return br;
|
for (total = 0; count; count -= br, total += br)
|
||||||
}
|
{
|
||||||
/* Finally, return the number of bytes read. */
|
br = dops->pread(dev, (char*) b + total, count, pos + total);
|
||||||
return total;
|
/* If everything ok, continue. */
|
||||||
|
if (br > 0) continue;
|
||||||
|
/* If EOF or error return number of bytes read. */
|
||||||
|
if (!br || total) return total;
|
||||||
|
/* Nothing read and error, return error status. */
|
||||||
|
return br;
|
||||||
|
}
|
||||||
|
/* Finally, return the number of bytes read. */
|
||||||
|
return total;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -223,46 +227,43 @@ s64 ntfs_pread(struct ntfs_device *dev, const s64 pos, s64 count, void *b)
|
|||||||
* appropriately to the return code of either seek, write, or set
|
* appropriately to the return code of either seek, write, or set
|
||||||
* to EINVAL in case of invalid arguments.
|
* to EINVAL in case of invalid arguments.
|
||||||
*/
|
*/
|
||||||
s64 ntfs_pwrite(struct ntfs_device *dev, const s64 pos, s64 count,
|
s64 ntfs_pwrite(struct ntfs_device *dev, const s64 pos, s64 count, const void *b)
|
||||||
const void *b)
|
|
||||||
{
|
{
|
||||||
s64 written, total, ret = -1;
|
s64 written, total, ret = -1;
|
||||||
struct ntfs_device_operations *dops;
|
struct ntfs_device_operations *dops;
|
||||||
|
|
||||||
ntfs_log_trace("pos %lld, count %lld\n",(long long)pos,(long long)count);
|
ntfs_log_trace("pos %lld, count %lld\n",(long long)pos,(long long)count);
|
||||||
|
|
||||||
if (!b || count < 0 || pos < 0) {
|
if (!b || count < 0 || pos < 0)
|
||||||
errno = EINVAL;
|
{
|
||||||
goto out;
|
errno = EINVAL;
|
||||||
}
|
goto out;
|
||||||
if (!count)
|
}
|
||||||
return 0;
|
if (!count) return 0;
|
||||||
if (NDevReadOnly(dev)) {
|
if (NDevReadOnly(dev))
|
||||||
errno = EROFS;
|
{
|
||||||
goto out;
|
errno = EROFS;
|
||||||
}
|
goto out;
|
||||||
|
}
|
||||||
dops = dev->d_ops;
|
|
||||||
|
|
||||||
NDevSetDirty(dev);
|
dops = dev->d_ops;
|
||||||
for (total = 0; count; count -= written, total += written) {
|
|
||||||
written = dops->pwrite(dev, (const char*)b + total, count,
|
NDevSetDirty(dev);
|
||||||
pos + total);
|
for (total = 0; count; count -= written, total += written)
|
||||||
/* If everything ok, continue. */
|
{
|
||||||
if (written > 0)
|
written = dops->pwrite(dev, (const char*) b + total, count, pos + total);
|
||||||
continue;
|
/* If everything ok, continue. */
|
||||||
/*
|
if (written > 0) continue;
|
||||||
* If nothing written or error return number of bytes written.
|
/*
|
||||||
*/
|
* If nothing written or error return number of bytes written.
|
||||||
if (!written || total)
|
*/
|
||||||
break;
|
if (!written || total) break;
|
||||||
/* Nothing written and error, return error status. */
|
/* Nothing written and error, return error status. */
|
||||||
total = written;
|
total = written;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
ret = total;
|
ret = total;
|
||||||
out:
|
out: return ret;
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -294,31 +295,29 @@ out:
|
|||||||
* sector transfer error. This should be detected by the caller by checking for
|
* sector transfer error. This should be detected by the caller by checking for
|
||||||
* the magic being "BAAD".
|
* the magic being "BAAD".
|
||||||
*/
|
*/
|
||||||
s64 ntfs_mst_pread(struct ntfs_device *dev, const s64 pos, s64 count,
|
s64 ntfs_mst_pread(struct ntfs_device *dev, const s64 pos, s64 count, const u32 bksize, void *b)
|
||||||
const u32 bksize, void *b)
|
|
||||||
{
|
{
|
||||||
s64 br, i;
|
s64 br, i;
|
||||||
|
|
||||||
if (bksize & (bksize - 1) || bksize % NTFS_BLOCK_SIZE) {
|
if (bksize & (bksize - 1) || bksize % NTFS_BLOCK_SIZE)
|
||||||
errno = EINVAL;
|
{
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
/* Do the read. */
|
}
|
||||||
br = ntfs_pread(dev, pos, count * bksize, b);
|
/* Do the read. */
|
||||||
if (br < 0)
|
br = ntfs_pread(dev, pos, count * bksize, b);
|
||||||
return br;
|
if (br < 0) return br;
|
||||||
/*
|
/*
|
||||||
* Apply fixups to successfully read data, disregarding any errors
|
* Apply fixups to successfully read data, disregarding any errors
|
||||||
* returned from the MST fixup function. This is because we want to
|
* returned from the MST fixup function. This is because we want to
|
||||||
* fixup everything possible and we rely on the fact that the "BAAD"
|
* fixup everything possible and we rely on the fact that the "BAAD"
|
||||||
* magic will be detected later on.
|
* magic will be detected later on.
|
||||||
*/
|
*/
|
||||||
count = br / bksize;
|
count = br / bksize;
|
||||||
for (i = 0; i < count; ++i)
|
for (i = 0; i < count; ++i)
|
||||||
ntfs_mst_post_read_fixup((NTFS_RECORD*)
|
ntfs_mst_post_read_fixup((NTFS_RECORD*) ((u8*) b + i * bksize), bksize);
|
||||||
((u8*)b + i * bksize), bksize);
|
/* Finally, return the number of complete blocks read. */
|
||||||
/* Finally, return the number of complete blocks read. */
|
return count;
|
||||||
return count;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -351,40 +350,38 @@ s64 ntfs_mst_pread(struct ntfs_device *dev, const s64 pos, s64 count,
|
|||||||
* simulating an mst read on the written data. This way cache coherency is
|
* simulating an mst read on the written data. This way cache coherency is
|
||||||
* achieved.
|
* achieved.
|
||||||
*/
|
*/
|
||||||
s64 ntfs_mst_pwrite(struct ntfs_device *dev, const s64 pos, s64 count,
|
s64 ntfs_mst_pwrite(struct ntfs_device *dev, const s64 pos, s64 count, const u32 bksize, void *b)
|
||||||
const u32 bksize, void *b)
|
|
||||||
{
|
{
|
||||||
s64 written, i;
|
s64 written, i;
|
||||||
|
|
||||||
if (count < 0 || bksize % NTFS_BLOCK_SIZE) {
|
if (count < 0 || bksize % NTFS_BLOCK_SIZE)
|
||||||
errno = EINVAL;
|
{
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
if (!count)
|
}
|
||||||
return 0;
|
if (!count) return 0;
|
||||||
/* Prepare data for writing. */
|
/* Prepare data for writing. */
|
||||||
for (i = 0; i < count; ++i) {
|
for (i = 0; i < count; ++i)
|
||||||
int err;
|
{
|
||||||
|
int err;
|
||||||
|
|
||||||
err = ntfs_mst_pre_write_fixup((NTFS_RECORD*)
|
err = ntfs_mst_pre_write_fixup((NTFS_RECORD*) ((u8*) b + i * bksize), bksize);
|
||||||
((u8*)b + i * bksize), bksize);
|
if (err < 0)
|
||||||
if (err < 0) {
|
{
|
||||||
/* Abort write at this position. */
|
/* Abort write at this position. */
|
||||||
if (!i)
|
if (!i) return err;
|
||||||
return err;
|
count = i;
|
||||||
count = i;
|
break;
|
||||||
break;
|
}
|
||||||
}
|
}
|
||||||
}
|
/* Write the prepared data. */
|
||||||
/* Write the prepared data. */
|
written = ntfs_pwrite(dev, pos, count * bksize, b);
|
||||||
written = ntfs_pwrite(dev, pos, count * bksize, b);
|
/* Quickly deprotect the data again. */
|
||||||
/* Quickly deprotect the data again. */
|
for (i = 0; i < count; ++i)
|
||||||
for (i = 0; i < count; ++i)
|
ntfs_mst_post_write_fixup((NTFS_RECORD*) ((u8*) b + i * bksize));
|
||||||
ntfs_mst_post_write_fixup((NTFS_RECORD*)((u8*)b + i * bksize));
|
if (written <= 0) return written;
|
||||||
if (written <= 0)
|
/* Finally, return the number of complete blocks written. */
|
||||||
return written;
|
return written / bksize;
|
||||||
/* Finally, return the number of complete blocks written. */
|
|
||||||
return written / bksize;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -398,29 +395,30 @@ s64 ntfs_mst_pwrite(struct ntfs_device *dev, const s64 pos, s64 count,
|
|||||||
* volume @vol into buffer @b. Return number of clusters read or -1 on error,
|
* volume @vol into buffer @b. Return number of clusters read or -1 on error,
|
||||||
* with errno set to the error code.
|
* with errno set to the error code.
|
||||||
*/
|
*/
|
||||||
s64 ntfs_cluster_read(const ntfs_volume *vol, const s64 lcn, const s64 count,
|
s64 ntfs_cluster_read(const ntfs_volume *vol, const s64 lcn, const s64 count, void *b)
|
||||||
void *b)
|
|
||||||
{
|
{
|
||||||
s64 br;
|
s64 br;
|
||||||
|
|
||||||
if (!vol || lcn < 0 || count < 0) {
|
if (!vol || lcn < 0 || count < 0)
|
||||||
errno = EINVAL;
|
{
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
if (vol->nr_clusters < lcn + count) {
|
}
|
||||||
errno = ESPIPE;
|
if (vol->nr_clusters < lcn + count)
|
||||||
ntfs_log_perror("Trying to read outside of volume "
|
{
|
||||||
"(%lld < %lld)", (long long)vol->nr_clusters,
|
errno = ESPIPE;
|
||||||
(long long)lcn + count);
|
ntfs_log_perror("Trying to read outside of volume "
|
||||||
return -1;
|
"(%lld < %lld)", (long long)vol->nr_clusters,
|
||||||
}
|
(long long)lcn + count);
|
||||||
br = ntfs_pread(vol->dev, lcn << vol->cluster_size_bits,
|
return -1;
|
||||||
count << vol->cluster_size_bits, b);
|
}
|
||||||
if (br < 0) {
|
br = ntfs_pread(vol->dev, lcn << vol->cluster_size_bits, count << vol->cluster_size_bits, b);
|
||||||
ntfs_log_perror("Error reading cluster(s)");
|
if (br < 0)
|
||||||
return br;
|
{
|
||||||
}
|
ntfs_log_perror("Error reading cluster(s)");
|
||||||
return br >> vol->cluster_size_bits;
|
return br;
|
||||||
|
}
|
||||||
|
return br >> vol->cluster_size_bits;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -434,32 +432,32 @@ s64 ntfs_cluster_read(const ntfs_volume *vol, const s64 lcn, const s64 count,
|
|||||||
* buffer @b to volume @vol. Return the number of clusters written or -1 on
|
* buffer @b to volume @vol. Return the number of clusters written or -1 on
|
||||||
* error, with errno set to the error code.
|
* error, with errno set to the error code.
|
||||||
*/
|
*/
|
||||||
s64 ntfs_cluster_write(const ntfs_volume *vol, const s64 lcn,
|
s64 ntfs_cluster_write(const ntfs_volume *vol, const s64 lcn, const s64 count, const void *b)
|
||||||
const s64 count, const void *b)
|
|
||||||
{
|
{
|
||||||
s64 bw;
|
s64 bw;
|
||||||
|
|
||||||
if (!vol || lcn < 0 || count < 0) {
|
if (!vol || lcn < 0 || count < 0)
|
||||||
errno = EINVAL;
|
{
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
if (vol->nr_clusters < lcn + count) {
|
}
|
||||||
errno = ESPIPE;
|
if (vol->nr_clusters < lcn + count)
|
||||||
ntfs_log_perror("Trying to write outside of volume "
|
{
|
||||||
"(%lld < %lld)", (long long)vol->nr_clusters,
|
errno = ESPIPE;
|
||||||
(long long)lcn + count);
|
ntfs_log_perror("Trying to write outside of volume "
|
||||||
return -1;
|
"(%lld < %lld)", (long long)vol->nr_clusters,
|
||||||
}
|
(long long)lcn + count);
|
||||||
if (!NVolReadOnly(vol))
|
return -1;
|
||||||
bw = ntfs_pwrite(vol->dev, lcn << vol->cluster_size_bits,
|
}
|
||||||
count << vol->cluster_size_bits, b);
|
if (!NVolReadOnly(vol))
|
||||||
else
|
bw = ntfs_pwrite(vol->dev, lcn << vol->cluster_size_bits, count << vol->cluster_size_bits, b);
|
||||||
bw = count << vol->cluster_size_bits;
|
else bw = count << vol->cluster_size_bits;
|
||||||
if (bw < 0) {
|
if (bw < 0)
|
||||||
ntfs_log_perror("Error writing cluster(s)");
|
{
|
||||||
return bw;
|
ntfs_log_perror("Error writing cluster(s)");
|
||||||
}
|
return bw;
|
||||||
return bw >> vol->cluster_size_bits;
|
}
|
||||||
|
return bw >> vol->cluster_size_bits;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -474,12 +472,10 @@ s64 ntfs_cluster_write(const ntfs_volume *vol, const s64 lcn,
|
|||||||
*/
|
*/
|
||||||
static int ntfs_device_offset_valid(struct ntfs_device *dev, s64 ofs)
|
static int ntfs_device_offset_valid(struct ntfs_device *dev, s64 ofs)
|
||||||
{
|
{
|
||||||
char ch;
|
char ch;
|
||||||
|
|
||||||
if (dev->d_ops->seek(dev, ofs, SEEK_SET) >= 0 &&
|
if (dev->d_ops->seek(dev, ofs, SEEK_SET) >= 0 && dev->d_ops->read(dev, &ch, 1) == 1) return 0;
|
||||||
dev->d_ops->read(dev, &ch, 1) == 1)
|
return -1;
|
||||||
return 0;
|
|
||||||
return -1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -496,61 +492,65 @@ static int ntfs_device_offset_valid(struct ntfs_device *dev, s64 ofs)
|
|||||||
*/
|
*/
|
||||||
s64 ntfs_device_size_get(struct ntfs_device *dev, int block_size)
|
s64 ntfs_device_size_get(struct ntfs_device *dev, int block_size)
|
||||||
{
|
{
|
||||||
s64 high, low;
|
s64 high, low;
|
||||||
|
|
||||||
if (!dev || block_size <= 0 || (block_size - 1) & block_size) {
|
if (!dev || block_size <= 0 || (block_size - 1) & block_size)
|
||||||
errno = EINVAL;
|
{
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
#ifdef BLKGETSIZE64
|
#ifdef BLKGETSIZE64
|
||||||
{ u64 size;
|
{ u64 size;
|
||||||
|
|
||||||
if (dev->d_ops->ioctl(dev, BLKGETSIZE64, &size) >= 0) {
|
if (dev->d_ops->ioctl(dev, BLKGETSIZE64, &size) >= 0)
|
||||||
ntfs_log_debug("BLKGETSIZE64 nr bytes = %llu (0x%llx)\n",
|
{
|
||||||
(unsigned long long)size,
|
ntfs_log_debug("BLKGETSIZE64 nr bytes = %llu (0x%llx)\n",
|
||||||
(unsigned long long)size);
|
(unsigned long long)size,
|
||||||
return (s64)size / block_size;
|
(unsigned long long)size);
|
||||||
}
|
return (s64)size / block_size;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef BLKGETSIZE
|
#ifdef BLKGETSIZE
|
||||||
{ unsigned long size;
|
{ unsigned long size;
|
||||||
|
|
||||||
if (dev->d_ops->ioctl(dev, BLKGETSIZE, &size) >= 0) {
|
if (dev->d_ops->ioctl(dev, BLKGETSIZE, &size) >= 0)
|
||||||
ntfs_log_debug("BLKGETSIZE nr 512 byte blocks = %lu (0x%lx)\n",
|
{
|
||||||
size, size);
|
ntfs_log_debug("BLKGETSIZE nr 512 byte blocks = %lu (0x%lx)\n",
|
||||||
return (s64)size * 512 / block_size;
|
size, size);
|
||||||
}
|
return (s64)size * 512 / block_size;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef FDGETPRM
|
#ifdef FDGETPRM
|
||||||
{ struct floppy_struct this_floppy;
|
{ struct floppy_struct this_floppy;
|
||||||
|
|
||||||
if (dev->d_ops->ioctl(dev, FDGETPRM, &this_floppy) >= 0) {
|
if (dev->d_ops->ioctl(dev, FDGETPRM, &this_floppy) >= 0)
|
||||||
ntfs_log_debug("FDGETPRM nr 512 byte blocks = %lu (0x%lx)\n",
|
{
|
||||||
(unsigned long)this_floppy.size,
|
ntfs_log_debug("FDGETPRM nr 512 byte blocks = %lu (0x%lx)\n",
|
||||||
(unsigned long)this_floppy.size);
|
(unsigned long)this_floppy.size,
|
||||||
return (s64)this_floppy.size * 512 / block_size;
|
(unsigned long)this_floppy.size);
|
||||||
}
|
return (s64)this_floppy.size * 512 / block_size;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
/*
|
/*
|
||||||
* We couldn't figure it out by using a specialized ioctl,
|
* We couldn't figure it out by using a specialized ioctl,
|
||||||
* so do binary search to find the size of the device.
|
* so do binary search to find the size of the device.
|
||||||
*/
|
*/
|
||||||
low = 0LL;
|
low = 0LL;
|
||||||
for (high = 1024LL; !ntfs_device_offset_valid(dev, high); high <<= 1)
|
for (high = 1024LL; !ntfs_device_offset_valid(dev, high); high <<= 1)
|
||||||
low = high;
|
low = high;
|
||||||
while (low < high - 1LL) {
|
while (low < high - 1LL)
|
||||||
const s64 mid = (low + high) / 2;
|
{
|
||||||
|
const s64 mid = (low + high) / 2;
|
||||||
|
|
||||||
if (!ntfs_device_offset_valid(dev, mid))
|
if (!ntfs_device_offset_valid(dev, mid))
|
||||||
low = mid;
|
low = mid;
|
||||||
else
|
else high = mid;
|
||||||
high = mid;
|
}
|
||||||
}
|
dev->d_ops->seek(dev, 0LL, SEEK_SET);
|
||||||
dev->d_ops->seek(dev, 0LL, SEEK_SET);
|
return (low + 1LL) / block_size;
|
||||||
return (low + 1LL) / block_size;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -567,23 +567,25 @@ s64 ntfs_device_size_get(struct ntfs_device *dev, int block_size)
|
|||||||
*/
|
*/
|
||||||
s64 ntfs_device_partition_start_sector_get(struct ntfs_device *dev)
|
s64 ntfs_device_partition_start_sector_get(struct ntfs_device *dev)
|
||||||
{
|
{
|
||||||
if (!dev) {
|
if (!dev)
|
||||||
errno = EINVAL;
|
{
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
#ifdef HDIO_GETGEO
|
#ifdef HDIO_GETGEO
|
||||||
{ struct hd_geometry geo;
|
{ struct hd_geometry geo;
|
||||||
|
|
||||||
if (!dev->d_ops->ioctl(dev, HDIO_GETGEO, &geo)) {
|
if (!dev->d_ops->ioctl(dev, HDIO_GETGEO, &geo))
|
||||||
ntfs_log_debug("HDIO_GETGEO start_sect = %lu (0x%lx)\n",
|
{
|
||||||
geo.start, geo.start);
|
ntfs_log_debug("HDIO_GETGEO start_sect = %lu (0x%lx)\n",
|
||||||
return geo.start;
|
geo.start, geo.start);
|
||||||
}
|
return geo.start;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
#else
|
#else
|
||||||
errno = EOPNOTSUPP;
|
errno = EOPNOTSUPP;
|
||||||
#endif
|
#endif
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -600,24 +602,26 @@ s64 ntfs_device_partition_start_sector_get(struct ntfs_device *dev)
|
|||||||
*/
|
*/
|
||||||
int ntfs_device_heads_get(struct ntfs_device *dev)
|
int ntfs_device_heads_get(struct ntfs_device *dev)
|
||||||
{
|
{
|
||||||
if (!dev) {
|
if (!dev)
|
||||||
errno = EINVAL;
|
{
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
#ifdef HDIO_GETGEO
|
#ifdef HDIO_GETGEO
|
||||||
{ struct hd_geometry geo;
|
{ struct hd_geometry geo;
|
||||||
|
|
||||||
if (!dev->d_ops->ioctl(dev, HDIO_GETGEO, &geo)) {
|
if (!dev->d_ops->ioctl(dev, HDIO_GETGEO, &geo))
|
||||||
ntfs_log_debug("HDIO_GETGEO heads = %u (0x%x)\n",
|
{
|
||||||
(unsigned)geo.heads,
|
ntfs_log_debug("HDIO_GETGEO heads = %u (0x%x)\n",
|
||||||
(unsigned)geo.heads);
|
(unsigned)geo.heads,
|
||||||
return geo.heads;
|
(unsigned)geo.heads);
|
||||||
}
|
return geo.heads;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
#else
|
#else
|
||||||
errno = EOPNOTSUPP;
|
errno = EOPNOTSUPP;
|
||||||
#endif
|
#endif
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -634,24 +638,26 @@ int ntfs_device_heads_get(struct ntfs_device *dev)
|
|||||||
*/
|
*/
|
||||||
int ntfs_device_sectors_per_track_get(struct ntfs_device *dev)
|
int ntfs_device_sectors_per_track_get(struct ntfs_device *dev)
|
||||||
{
|
{
|
||||||
if (!dev) {
|
if (!dev)
|
||||||
errno = EINVAL;
|
{
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
#ifdef HDIO_GETGEO
|
#ifdef HDIO_GETGEO
|
||||||
{ struct hd_geometry geo;
|
{ struct hd_geometry geo;
|
||||||
|
|
||||||
if (!dev->d_ops->ioctl(dev, HDIO_GETGEO, &geo)) {
|
if (!dev->d_ops->ioctl(dev, HDIO_GETGEO, &geo))
|
||||||
ntfs_log_debug("HDIO_GETGEO sectors_per_track = %u (0x%x)\n",
|
{
|
||||||
(unsigned)geo.sectors,
|
ntfs_log_debug("HDIO_GETGEO sectors_per_track = %u (0x%x)\n",
|
||||||
(unsigned)geo.sectors);
|
(unsigned)geo.sectors,
|
||||||
return geo.sectors;
|
(unsigned)geo.sectors);
|
||||||
}
|
return geo.sectors;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
#else
|
#else
|
||||||
errno = EOPNOTSUPP;
|
errno = EOPNOTSUPP;
|
||||||
#endif
|
#endif
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -668,24 +674,26 @@ int ntfs_device_sectors_per_track_get(struct ntfs_device *dev)
|
|||||||
*/
|
*/
|
||||||
int ntfs_device_sector_size_get(struct ntfs_device *dev)
|
int ntfs_device_sector_size_get(struct ntfs_device *dev)
|
||||||
{
|
{
|
||||||
if (!dev) {
|
if (!dev)
|
||||||
errno = EINVAL;
|
{
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
#ifdef BLKSSZGET
|
#ifdef BLKSSZGET
|
||||||
{
|
{
|
||||||
int sect_size = 0;
|
int sect_size = 0;
|
||||||
|
|
||||||
if (!dev->d_ops->ioctl(dev, BLKSSZGET, §_size)) {
|
if (!dev->d_ops->ioctl(dev, BLKSSZGET, §_size))
|
||||||
ntfs_log_debug("BLKSSZGET sector size = %d bytes\n",
|
{
|
||||||
sect_size);
|
ntfs_log_debug("BLKSSZGET sector size = %d bytes\n",
|
||||||
return sect_size;
|
sect_size);
|
||||||
}
|
return sect_size;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
#else
|
#else
|
||||||
errno = EOPNOTSUPP;
|
errno = EOPNOTSUPP;
|
||||||
#endif
|
#endif
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -701,30 +709,30 @@ int ntfs_device_sector_size_get(struct ntfs_device *dev)
|
|||||||
* EOPNOTSUPP System does not support BLKBSZSET ioctl
|
* EOPNOTSUPP System does not support BLKBSZSET ioctl
|
||||||
* ENOTTY @dev is a file or a device not supporting BLKBSZSET
|
* ENOTTY @dev is a file or a device not supporting BLKBSZSET
|
||||||
*/
|
*/
|
||||||
int ntfs_device_block_size_set(struct ntfs_device *dev,
|
int ntfs_device_block_size_set(struct ntfs_device *dev, int block_size __attribute__((unused)))
|
||||||
int block_size __attribute__((unused)))
|
|
||||||
{
|
{
|
||||||
if (!dev) {
|
if (!dev)
|
||||||
errno = EINVAL;
|
{
|
||||||
return -1;
|
errno = EINVAL;
|
||||||
}
|
return -1;
|
||||||
|
}
|
||||||
#ifdef BLKBSZSET
|
#ifdef BLKBSZSET
|
||||||
{
|
{
|
||||||
size_t s_block_size = block_size;
|
size_t s_block_size = block_size;
|
||||||
if (!dev->d_ops->ioctl(dev, BLKBSZSET, &s_block_size)) {
|
if (!dev->d_ops->ioctl(dev, BLKBSZSET, &s_block_size))
|
||||||
ntfs_log_debug("Used BLKBSZSET to set block size to "
|
{
|
||||||
"%d bytes.\n", block_size);
|
ntfs_log_debug("Used BLKBSZSET to set block size to "
|
||||||
return 0;
|
"%d bytes.\n", block_size);
|
||||||
}
|
return 0;
|
||||||
/* If not a block device, pretend it was successful. */
|
}
|
||||||
if (!NDevBlock(dev))
|
/* If not a block device, pretend it was successful. */
|
||||||
return 0;
|
if (!NDevBlock(dev))
|
||||||
}
|
return 0;
|
||||||
|
}
|
||||||
#else
|
#else
|
||||||
/* If not a block device, pretend it was successful. */
|
/* If not a block device, pretend it was successful. */
|
||||||
if (!NDevBlock(dev))
|
if (!NDevBlock(dev)) return 0;
|
||||||
return 0;
|
errno = EOPNOTSUPP;
|
||||||
errno = EOPNOTSUPP;
|
|
||||||
#endif
|
#endif
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
@ -36,11 +36,13 @@
|
|||||||
*
|
*
|
||||||
* Defined bits for the state field in the ntfs_device structure.
|
* Defined bits for the state field in the ntfs_device structure.
|
||||||
*/
|
*/
|
||||||
typedef enum {
|
typedef enum
|
||||||
ND_Open, /* 1: Device is open. */
|
{
|
||||||
ND_ReadOnly, /* 1: Device is read-only. */
|
ND_Open, /* 1: Device is open. */
|
||||||
ND_Dirty, /* 1: Device is dirty, needs sync. */
|
ND_ReadOnly, /* 1: Device is read-only. */
|
||||||
ND_Block, /* 1: Device is a block device. */
|
ND_Dirty, /* 1: Device is dirty, needs sync. */
|
||||||
|
ND_Block,
|
||||||
|
/* 1: Device is a block device. */
|
||||||
} ntfs_device_state_bits;
|
} ntfs_device_state_bits;
|
||||||
|
|
||||||
#define test_ndev_flag(nd, flag) test_bit(ND_##flag, (nd)->d_state)
|
#define test_ndev_flag(nd, flag) test_bit(ND_##flag, (nd)->d_state)
|
||||||
@ -69,12 +71,13 @@ typedef enum {
|
|||||||
* The ntfs device structure defining all operations needed to access the low
|
* The ntfs device structure defining all operations needed to access the low
|
||||||
* level device underlying the ntfs volume.
|
* level device underlying the ntfs volume.
|
||||||
*/
|
*/
|
||||||
struct ntfs_device {
|
struct ntfs_device
|
||||||
struct ntfs_device_operations *d_ops; /* Device operations. */
|
{
|
||||||
unsigned long d_state; /* State of the device. */
|
struct ntfs_device_operations *d_ops; /* Device operations. */
|
||||||
char *d_name; /* Name of device. */
|
unsigned long d_state; /* State of the device. */
|
||||||
void *d_private; /* Private data used by the
|
char *d_name; /* Name of device. */
|
||||||
device operations. */
|
void *d_private; /* Private data used by the
|
||||||
|
device operations. */
|
||||||
};
|
};
|
||||||
|
|
||||||
struct stat;
|
struct stat;
|
||||||
@ -85,38 +88,32 @@ struct stat;
|
|||||||
* The ntfs device operations defining all operations that can be performed on
|
* The ntfs device operations defining all operations that can be performed on
|
||||||
* the low level device described by an ntfs device structure.
|
* the low level device described by an ntfs device structure.
|
||||||
*/
|
*/
|
||||||
struct ntfs_device_operations {
|
struct ntfs_device_operations
|
||||||
int (*open)(struct ntfs_device *dev, int flags);
|
{
|
||||||
int (*close)(struct ntfs_device *dev);
|
int (*open)(struct ntfs_device *dev, int flags);
|
||||||
s64 (*seek)(struct ntfs_device *dev, s64 offset, int whence);
|
int (*close)(struct ntfs_device *dev);
|
||||||
s64 (*read)(struct ntfs_device *dev, void *buf, s64 count);
|
s64 (*seek)(struct ntfs_device *dev, s64 offset, int whence);
|
||||||
s64 (*write)(struct ntfs_device *dev, const void *buf, s64 count);
|
s64 (*read)(struct ntfs_device *dev, void *buf, s64 count);
|
||||||
s64 (*pread)(struct ntfs_device *dev, void *buf, s64 count, s64 offset);
|
s64 (*write)(struct ntfs_device *dev, const void *buf, s64 count);
|
||||||
s64 (*pwrite)(struct ntfs_device *dev, const void *buf, s64 count,
|
s64 (*pread)(struct ntfs_device *dev, void *buf, s64 count, s64 offset);
|
||||||
s64 offset);
|
s64 (*pwrite)(struct ntfs_device *dev, const void *buf, s64 count, s64 offset);
|
||||||
int (*sync)(struct ntfs_device *dev);
|
int (*sync)(struct ntfs_device *dev);
|
||||||
int (*stat)(struct ntfs_device *dev, struct stat *buf);
|
int (*stat)(struct ntfs_device *dev, struct stat *buf);
|
||||||
int (*ioctl)(struct ntfs_device *dev, int request, void *argp);
|
int (*ioctl)(struct ntfs_device *dev, int request, void *argp);
|
||||||
};
|
};
|
||||||
|
|
||||||
extern struct ntfs_device *ntfs_device_alloc(const char *name, const long state,
|
extern struct ntfs_device *ntfs_device_alloc(const char *name, const long state, struct ntfs_device_operations *dops,
|
||||||
struct ntfs_device_operations *dops, void *priv_data);
|
void *priv_data);
|
||||||
extern int ntfs_device_free(struct ntfs_device *dev);
|
extern int ntfs_device_free(struct ntfs_device *dev);
|
||||||
|
|
||||||
extern s64 ntfs_pread(struct ntfs_device *dev, const s64 pos, s64 count,
|
extern s64 ntfs_pread(struct ntfs_device *dev, const s64 pos, s64 count, void *b);
|
||||||
void *b);
|
extern s64 ntfs_pwrite(struct ntfs_device *dev, const s64 pos, s64 count, const void *b);
|
||||||
extern s64 ntfs_pwrite(struct ntfs_device *dev, const s64 pos, s64 count,
|
|
||||||
const void *b);
|
|
||||||
|
|
||||||
extern s64 ntfs_mst_pread(struct ntfs_device *dev, const s64 pos, s64 count,
|
extern s64 ntfs_mst_pread(struct ntfs_device *dev, const s64 pos, s64 count, const u32 bksize, void *b);
|
||||||
const u32 bksize, void *b);
|
extern s64 ntfs_mst_pwrite(struct ntfs_device *dev, const s64 pos, s64 count, const u32 bksize, void *b);
|
||||||
extern s64 ntfs_mst_pwrite(struct ntfs_device *dev, const s64 pos, s64 count,
|
|
||||||
const u32 bksize, void *b);
|
|
||||||
|
|
||||||
extern s64 ntfs_cluster_read(const ntfs_volume *vol, const s64 lcn,
|
extern s64 ntfs_cluster_read(const ntfs_volume *vol, const s64 lcn, const s64 count, void *b);
|
||||||
const s64 count, void *b);
|
extern s64 ntfs_cluster_write(const ntfs_volume *vol, const s64 lcn, const s64 count, const void *b);
|
||||||
extern s64 ntfs_cluster_write(const ntfs_volume *vol, const s64 lcn,
|
|
||||||
const s64 count, const void *b);
|
|
||||||
|
|
||||||
extern s64 ntfs_device_size_get(struct ntfs_device *dev, int block_size);
|
extern s64 ntfs_device_size_get(struct ntfs_device *dev, int block_size);
|
||||||
extern s64 ntfs_device_partition_start_sector_get(struct ntfs_device *dev);
|
extern s64 ntfs_device_partition_start_sector_get(struct ntfs_device *dev);
|
||||||
|
@ -45,11 +45,12 @@
|
|||||||
/**
|
/**
|
||||||
* struct hd_geometry -
|
* struct hd_geometry -
|
||||||
*/
|
*/
|
||||||
struct hd_geometry {
|
struct hd_geometry
|
||||||
unsigned char heads;
|
{
|
||||||
unsigned char sectors;
|
unsigned char heads;
|
||||||
unsigned short cylinders;
|
unsigned char sectors;
|
||||||
unsigned long start;
|
unsigned short cylinders;
|
||||||
|
unsigned long start;
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
#ifndef BLKGETSIZE
|
#ifndef BLKGETSIZE
|
||||||
@ -70,7 +71,6 @@ struct hd_geometry {
|
|||||||
|
|
||||||
#endif /* __CYGWIN32__ */
|
#endif /* __CYGWIN32__ */
|
||||||
|
|
||||||
|
|
||||||
/* Forward declaration. */
|
/* Forward declaration. */
|
||||||
struct ntfs_device_operations;
|
struct ntfs_device_operations;
|
||||||
|
|
||||||
|
4149
source/libntfs/dir.c
4149
source/libntfs/dir.c
File diff suppressed because it is too large
Load Diff
@ -59,27 +59,21 @@ extern ntfschar NTFS_INDEX_O[3];
|
|||||||
extern ntfschar NTFS_INDEX_Q[3];
|
extern ntfschar NTFS_INDEX_Q[3];
|
||||||
extern ntfschar NTFS_INDEX_R[3];
|
extern ntfschar NTFS_INDEX_R[3];
|
||||||
|
|
||||||
extern u64 ntfs_inode_lookup_by_name(ntfs_inode *dir_ni,
|
extern u64 ntfs_inode_lookup_by_name(ntfs_inode *dir_ni, const ntfschar *uname, const int uname_len);
|
||||||
const ntfschar *uname, const int uname_len);
|
|
||||||
extern u64 ntfs_inode_lookup_by_mbsname(ntfs_inode *dir_ni, const char *name);
|
extern u64 ntfs_inode_lookup_by_mbsname(ntfs_inode *dir_ni, const char *name);
|
||||||
extern void ntfs_inode_update_mbsname(ntfs_inode *dir_ni, const char *name,
|
extern void ntfs_inode_update_mbsname(ntfs_inode *dir_ni, const char *name, u64 inum);
|
||||||
u64 inum);
|
|
||||||
|
|
||||||
extern ntfs_inode *ntfs_pathname_to_inode(ntfs_volume *vol, ntfs_inode *parent,
|
extern ntfs_inode *ntfs_pathname_to_inode(ntfs_volume *vol, ntfs_inode *parent, const char *pathname);
|
||||||
const char *pathname);
|
extern ntfs_inode *ntfs_create(ntfs_inode *dir_ni, le32 securid, ntfschar *name, u8 name_len, mode_t type);
|
||||||
extern ntfs_inode *ntfs_create(ntfs_inode *dir_ni, le32 securid,
|
extern ntfs_inode *ntfs_create_device(ntfs_inode *dir_ni, le32 securid, ntfschar *name, u8 name_len, mode_t type,
|
||||||
ntfschar *name, u8 name_len, mode_t type);
|
dev_t dev);
|
||||||
extern ntfs_inode *ntfs_create_device(ntfs_inode *dir_ni, le32 securid,
|
extern ntfs_inode *ntfs_create_symlink(ntfs_inode *dir_ni, le32 securid, ntfschar *name, u8 name_len, ntfschar *target,
|
||||||
ntfschar *name, u8 name_len, mode_t type, dev_t dev);
|
int target_len);
|
||||||
extern ntfs_inode *ntfs_create_symlink(ntfs_inode *dir_ni, le32 securid,
|
|
||||||
ntfschar *name, u8 name_len, ntfschar *target, int target_len);
|
|
||||||
extern int ntfs_check_empty_dir(ntfs_inode *ni);
|
extern int ntfs_check_empty_dir(ntfs_inode *ni);
|
||||||
extern int ntfs_delete(ntfs_volume *vol, const char *path,
|
extern int ntfs_delete(ntfs_volume *vol, const char *path, ntfs_inode *ni, ntfs_inode *dir_ni, ntfschar *name,
|
||||||
ntfs_inode *ni, ntfs_inode *dir_ni, ntfschar *name,
|
u8 name_len);
|
||||||
u8 name_len);
|
|
||||||
|
|
||||||
extern int ntfs_link(ntfs_inode *ni, ntfs_inode *dir_ni, ntfschar *name,
|
extern int ntfs_link(ntfs_inode *ni, ntfs_inode *dir_ni, ntfschar *name, u8 name_len);
|
||||||
u8 name_len);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* File types (adapted from include <linux/fs.h>)
|
* File types (adapted from include <linux/fs.h>)
|
||||||
@ -100,19 +94,15 @@ extern int ntfs_link(ntfs_inode *ni, ntfs_inode *dir_ni, ntfschar *name,
|
|||||||
* This allows the caller to read directories into their application or
|
* This allows the caller to read directories into their application or
|
||||||
* to have different dirent layouts depending on the binary type.
|
* to have different dirent layouts depending on the binary type.
|
||||||
*/
|
*/
|
||||||
typedef int (*ntfs_filldir_t)(void *dirent, const ntfschar *name,
|
typedef int (*ntfs_filldir_t)(void *dirent, const ntfschar *name, const int name_len, const int name_type,
|
||||||
const int name_len, const int name_type, const s64 pos,
|
const s64 pos, const MFT_REF mref, const unsigned dt_type);
|
||||||
const MFT_REF mref, const unsigned dt_type);
|
|
||||||
|
|
||||||
extern int ntfs_readdir(ntfs_inode *dir_ni, s64 *pos,
|
extern int ntfs_readdir(ntfs_inode *dir_ni, s64 *pos, void *dirent, ntfs_filldir_t filldir);
|
||||||
void *dirent, ntfs_filldir_t filldir);
|
|
||||||
|
|
||||||
ntfs_inode *ntfs_dir_parent_inode(ntfs_inode *ni);
|
ntfs_inode *ntfs_dir_parent_inode(ntfs_inode *ni);
|
||||||
|
|
||||||
int ntfs_get_ntfs_dos_name(ntfs_inode *ni, ntfs_inode *dir_ni,
|
int ntfs_get_ntfs_dos_name(ntfs_inode *ni, ntfs_inode *dir_ni, char *value, size_t size);
|
||||||
char *value, size_t size);
|
int ntfs_set_ntfs_dos_name(ntfs_inode *ni, ntfs_inode *dir_ni, const char *value, size_t size, int flags);
|
||||||
int ntfs_set_ntfs_dos_name(ntfs_inode *ni, ntfs_inode *dir_ni,
|
|
||||||
const char *value, size_t size, int flags);
|
|
||||||
int ntfs_remove_ntfs_dos_name(ntfs_inode *ni, ntfs_inode *dir_ni);
|
int ntfs_remove_ntfs_dos_name(ntfs_inode *ni, ntfs_inode *dir_ni);
|
||||||
|
|
||||||
#if CACHE_INODE_SIZE
|
#if CACHE_INODE_SIZE
|
||||||
|
@ -60,14 +60,14 @@
|
|||||||
|
|
||||||
#ifdef HAVE_SETXATTR /* extended attributes interface required */
|
#ifdef HAVE_SETXATTR /* extended attributes interface required */
|
||||||
|
|
||||||
static ntfschar logged_utility_stream_name[] = {
|
static ntfschar logged_utility_stream_name[] =
|
||||||
const_cpu_to_le16('$'),
|
{
|
||||||
const_cpu_to_le16('E'),
|
const_cpu_to_le16('$'),
|
||||||
const_cpu_to_le16('F'),
|
const_cpu_to_le16('E'),
|
||||||
const_cpu_to_le16('S'),
|
const_cpu_to_le16('F'),
|
||||||
const_cpu_to_le16(0)
|
const_cpu_to_le16('S'),
|
||||||
} ;
|
const_cpu_to_le16(0)
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Get the ntfs EFS info into an extended attribute
|
* Get the ntfs EFS info into an extended attribute
|
||||||
@ -75,49 +75,63 @@ static ntfschar logged_utility_stream_name[] = {
|
|||||||
|
|
||||||
int ntfs_get_efs_info(ntfs_inode *ni, char *value, size_t size)
|
int ntfs_get_efs_info(ntfs_inode *ni, char *value, size_t size)
|
||||||
{
|
{
|
||||||
EFS_ATTR_HEADER *efs_info;
|
EFS_ATTR_HEADER *efs_info;
|
||||||
s64 attr_size = 0;
|
s64 attr_size = 0;
|
||||||
|
|
||||||
if (ni) {
|
if (ni)
|
||||||
if (ni->flags & FILE_ATTR_ENCRYPTED) {
|
{
|
||||||
efs_info = (EFS_ATTR_HEADER*)ntfs_attr_readall(ni,
|
if (ni->flags & FILE_ATTR_ENCRYPTED)
|
||||||
AT_LOGGED_UTILITY_STREAM,(ntfschar*)NULL, 0,
|
{
|
||||||
&attr_size);
|
efs_info = (EFS_ATTR_HEADER*)ntfs_attr_readall(ni,
|
||||||
if (efs_info
|
AT_LOGGED_UTILITY_STREAM,(ntfschar*)NULL, 0,
|
||||||
&& (le32_to_cpu(efs_info->length) == attr_size)) {
|
&attr_size);
|
||||||
if (attr_size <= (s64)size) {
|
if (efs_info
|
||||||
if (value)
|
&& (le32_to_cpu(efs_info->length) == attr_size))
|
||||||
memcpy(value,efs_info,attr_size);
|
{
|
||||||
else {
|
if (attr_size <= (s64)size)
|
||||||
errno = EFAULT;
|
{
|
||||||
attr_size = 0;
|
if (value)
|
||||||
}
|
memcpy(value,efs_info,attr_size);
|
||||||
} else
|
else
|
||||||
if (size) {
|
{
|
||||||
errno = ERANGE;
|
errno = EFAULT;
|
||||||
attr_size = 0;
|
attr_size = 0;
|
||||||
}
|
}
|
||||||
free (efs_info);
|
}
|
||||||
} else {
|
else
|
||||||
if (efs_info) {
|
if (size)
|
||||||
free(efs_info);
|
{
|
||||||
ntfs_log_error("Bad efs_info for inode %lld\n",
|
errno = ERANGE;
|
||||||
(long long)ni->mft_no);
|
attr_size = 0;
|
||||||
} else {
|
}
|
||||||
ntfs_log_error("Could not get efsinfo"
|
free (efs_info);
|
||||||
" for inode %lld\n",
|
}
|
||||||
(long long)ni->mft_no);
|
else
|
||||||
}
|
{
|
||||||
errno = EIO;
|
if (efs_info)
|
||||||
attr_size = 0;
|
{
|
||||||
}
|
free(efs_info);
|
||||||
} else {
|
ntfs_log_error("Bad efs_info for inode %lld\n",
|
||||||
errno = ENODATA;
|
(long long)ni->mft_no);
|
||||||
ntfs_log_trace("Inode %lld is not encrypted\n",
|
}
|
||||||
(long long)ni->mft_no);
|
else
|
||||||
}
|
{
|
||||||
}
|
ntfs_log_error("Could not get efsinfo"
|
||||||
return (attr_size ? (int)attr_size : -errno);
|
" for inode %lld\n",
|
||||||
|
(long long)ni->mft_no);
|
||||||
|
}
|
||||||
|
errno = EIO;
|
||||||
|
attr_size = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
errno = ENODATA;
|
||||||
|
ntfs_log_trace("Inode %lld is not encrypted\n",
|
||||||
|
(long long)ni->mft_no);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return (attr_size ? (int)attr_size : -errno);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -135,76 +149,89 @@ int ntfs_get_efs_info(ntfs_inode *ni, char *value, size_t size)
|
|||||||
|
|
||||||
static int fixup_loop(ntfs_inode *ni)
|
static int fixup_loop(ntfs_inode *ni)
|
||||||
{
|
{
|
||||||
ntfs_attr_search_ctx *ctx;
|
ntfs_attr_search_ctx *ctx;
|
||||||
ntfs_attr *na;
|
ntfs_attr *na;
|
||||||
ATTR_RECORD *a;
|
ATTR_RECORD *a;
|
||||||
BOOL restart;
|
BOOL restart;
|
||||||
BOOL first;
|
BOOL first;
|
||||||
int cnt;
|
int cnt;
|
||||||
int maxcnt;
|
int maxcnt;
|
||||||
int res = 0;
|
int res = 0;
|
||||||
|
|
||||||
maxcnt = 0;
|
maxcnt = 0;
|
||||||
do {
|
do
|
||||||
restart = FALSE;
|
{
|
||||||
ctx = ntfs_attr_get_search_ctx(ni, NULL);
|
restart = FALSE;
|
||||||
if (!ctx) {
|
ctx = ntfs_attr_get_search_ctx(ni, NULL);
|
||||||
ntfs_log_error("Failed to get ctx for efs\n");
|
if (!ctx)
|
||||||
res = -1;
|
{
|
||||||
}
|
ntfs_log_error("Failed to get ctx for efs\n");
|
||||||
cnt = 0;
|
res = -1;
|
||||||
while (!restart && !res
|
}
|
||||||
&& !ntfs_attr_lookup(AT_DATA, NULL, 0,
|
cnt = 0;
|
||||||
CASE_SENSITIVE, 0, NULL, 0, ctx)) {
|
while (!restart && !res
|
||||||
cnt++;
|
&& !ntfs_attr_lookup(AT_DATA, NULL, 0,
|
||||||
a = ctx->attr;
|
CASE_SENSITIVE, 0, NULL, 0, ctx))
|
||||||
na = ntfs_attr_open(ctx->ntfs_ino, AT_DATA,
|
{
|
||||||
(ntfschar*)((u8*)a + le16_to_cpu(a->name_offset)),
|
cnt++;
|
||||||
a->name_length);
|
a = ctx->attr;
|
||||||
if (!na) {
|
na = ntfs_attr_open(ctx->ntfs_ino, AT_DATA,
|
||||||
ntfs_log_error("can't open DATA Attribute\n");
|
(ntfschar*)((u8*)a + le16_to_cpu(a->name_offset)),
|
||||||
res = -1;
|
a->name_length);
|
||||||
}
|
if (!na)
|
||||||
if (na && !(ctx->attr->flags & ATTR_IS_ENCRYPTED)) {
|
{
|
||||||
if (!NAttrNonResident(na)
|
ntfs_log_error("can't open DATA Attribute\n");
|
||||||
&& ntfs_attr_make_non_resident(na, ctx)) {
|
res = -1;
|
||||||
/*
|
}
|
||||||
* ntfs_attr_make_non_resident fails if there
|
if (na && !(ctx->attr->flags & ATTR_IS_ENCRYPTED))
|
||||||
* is not enough space in the MFT record.
|
{
|
||||||
* When this happens, force making non-resident
|
if (!NAttrNonResident(na)
|
||||||
* so that some other attribute is expelled.
|
&& ntfs_attr_make_non_resident(na, ctx))
|
||||||
*/
|
{
|
||||||
if (ntfs_attr_force_non_resident(na)) {
|
/*
|
||||||
res = -1;
|
* ntfs_attr_make_non_resident fails if there
|
||||||
} else {
|
* is not enough space in the MFT record.
|
||||||
/* make sure there is some progress */
|
* When this happens, force making non-resident
|
||||||
if (cnt <= maxcnt) {
|
* so that some other attribute is expelled.
|
||||||
errno = EIO;
|
*/
|
||||||
ntfs_log_error("Multiple failure"
|
if (ntfs_attr_force_non_resident(na))
|
||||||
" making non resident\n");
|
{
|
||||||
res = -1;
|
res = -1;
|
||||||
} else {
|
}
|
||||||
ntfs_attr_put_search_ctx(ctx);
|
else
|
||||||
ctx = (ntfs_attr_search_ctx*)NULL;
|
{
|
||||||
restart = TRUE;
|
/* make sure there is some progress */
|
||||||
maxcnt = cnt;
|
if (cnt <= maxcnt)
|
||||||
}
|
{
|
||||||
}
|
errno = EIO;
|
||||||
}
|
ntfs_log_error("Multiple failure"
|
||||||
if (!restart && !res
|
" making non resident\n");
|
||||||
&& ntfs_efs_fixup_attribute(ctx, na)) {
|
res = -1;
|
||||||
ntfs_log_error("Error in efs fixup of AT_DATA Attribute\n");
|
}
|
||||||
res = -1;
|
else
|
||||||
}
|
{
|
||||||
}
|
ntfs_attr_put_search_ctx(ctx);
|
||||||
if (na)
|
ctx = (ntfs_attr_search_ctx*)NULL;
|
||||||
ntfs_attr_close(na);
|
restart = TRUE;
|
||||||
}
|
maxcnt = cnt;
|
||||||
first = FALSE;
|
}
|
||||||
} while (restart && !res);
|
}
|
||||||
if (ctx)
|
}
|
||||||
ntfs_attr_put_search_ctx(ctx);
|
if (!restart && !res
|
||||||
return (res);
|
&& ntfs_efs_fixup_attribute(ctx, na))
|
||||||
|
{
|
||||||
|
ntfs_log_error("Error in efs fixup of AT_DATA Attribute\n");
|
||||||
|
res = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (na)
|
||||||
|
ntfs_attr_close(na);
|
||||||
|
}
|
||||||
|
first = FALSE;
|
||||||
|
}while (restart && !res);
|
||||||
|
if (ctx)
|
||||||
|
ntfs_attr_put_search_ctx(ctx);
|
||||||
|
return (res);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -214,100 +241,121 @@ static int fixup_loop(ntfs_inode *ni)
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
int ntfs_set_efs_info(ntfs_inode *ni, const char *value, size_t size,
|
int ntfs_set_efs_info(ntfs_inode *ni, const char *value, size_t size,
|
||||||
int flags)
|
int flags)
|
||||||
|
|
||||||
{
|
|
||||||
int res;
|
|
||||||
int written;
|
|
||||||
ntfs_attr *na;
|
|
||||||
const EFS_ATTR_HEADER *info_header;
|
|
||||||
|
|
||||||
res = 0;
|
{
|
||||||
if (ni && value && size) {
|
int res;
|
||||||
if (ni->flags & (FILE_ATTR_ENCRYPTED | FILE_ATTR_COMPRESSED)) {
|
int written;
|
||||||
if (ni->flags & FILE_ATTR_ENCRYPTED) {
|
ntfs_attr *na;
|
||||||
ntfs_log_trace("Inode %lld already encrypted\n",
|
const EFS_ATTR_HEADER *info_header;
|
||||||
(long long)ni->mft_no);
|
|
||||||
errno = EEXIST;
|
res = 0;
|
||||||
} else {
|
if (ni && value && size)
|
||||||
/*
|
{
|
||||||
* Possible problem : if encrypted file was
|
if (ni->flags & (FILE_ATTR_ENCRYPTED | FILE_ATTR_COMPRESSED))
|
||||||
* restored in a compressed directory, it was
|
{
|
||||||
* restored as compressed.
|
if (ni->flags & FILE_ATTR_ENCRYPTED)
|
||||||
* TODO : decompress first.
|
{
|
||||||
*/
|
ntfs_log_trace("Inode %lld already encrypted\n",
|
||||||
ntfs_log_error("Inode %lld cannot be encrypted and compressed\n",
|
(long long)ni->mft_no);
|
||||||
(long long)ni->mft_no);
|
errno = EEXIST;
|
||||||
errno = EIO;
|
}
|
||||||
}
|
else
|
||||||
return -1;
|
{
|
||||||
}
|
/*
|
||||||
info_header = (const EFS_ATTR_HEADER*)value;
|
* Possible problem : if encrypted file was
|
||||||
/* make sure we get a likely efsinfo */
|
* restored in a compressed directory, it was
|
||||||
if (le32_to_cpu(info_header->length) != size) {
|
* restored as compressed.
|
||||||
errno = EINVAL;
|
* TODO : decompress first.
|
||||||
return (-1);
|
*/
|
||||||
}
|
ntfs_log_error("Inode %lld cannot be encrypted and compressed\n",
|
||||||
if (!ntfs_attr_exist(ni,AT_LOGGED_UTILITY_STREAM,
|
(long long)ni->mft_no);
|
||||||
(ntfschar*)NULL,0)) {
|
errno = EIO;
|
||||||
if (!(flags & XATTR_REPLACE)) {
|
}
|
||||||
/*
|
return -1;
|
||||||
* no logged_utility_stream attribute : add one,
|
}
|
||||||
* apparently, this does not feed the new value in
|
info_header = (const EFS_ATTR_HEADER*)value;
|
||||||
*/
|
/* make sure we get a likely efsinfo */
|
||||||
res = ntfs_attr_add(ni,AT_LOGGED_UTILITY_STREAM,
|
if (le32_to_cpu(info_header->length) != size)
|
||||||
logged_utility_stream_name,4,
|
{
|
||||||
(u8*)NULL,(s64)size);
|
errno = EINVAL;
|
||||||
} else {
|
return (-1);
|
||||||
errno = ENODATA;
|
}
|
||||||
res = -1;
|
if (!ntfs_attr_exist(ni,AT_LOGGED_UTILITY_STREAM,
|
||||||
}
|
(ntfschar*)NULL,0))
|
||||||
} else {
|
{
|
||||||
errno = EEXIST;
|
if (!(flags & XATTR_REPLACE))
|
||||||
res = -1;
|
{
|
||||||
}
|
/*
|
||||||
if (!res) {
|
* no logged_utility_stream attribute : add one,
|
||||||
/*
|
* apparently, this does not feed the new value in
|
||||||
* open and update the existing efs data
|
*/
|
||||||
*/
|
res = ntfs_attr_add(ni,AT_LOGGED_UTILITY_STREAM,
|
||||||
na = ntfs_attr_open(ni, AT_LOGGED_UTILITY_STREAM,
|
logged_utility_stream_name,4,
|
||||||
logged_utility_stream_name, 4);
|
(u8*)NULL,(s64)size);
|
||||||
if (na) {
|
}
|
||||||
/* resize attribute */
|
else
|
||||||
res = ntfs_attr_truncate(na, (s64)size);
|
{
|
||||||
/* overwrite value if any */
|
errno = ENODATA;
|
||||||
if (!res && value) {
|
res = -1;
|
||||||
written = (int)ntfs_attr_pwrite(na,
|
}
|
||||||
(s64)0, (s64)size, value);
|
}
|
||||||
if (written != (s64)size) {
|
else
|
||||||
ntfs_log_error("Failed to "
|
{
|
||||||
"update efs data\n");
|
errno = EEXIST;
|
||||||
errno = EIO;
|
res = -1;
|
||||||
res = -1;
|
}
|
||||||
}
|
if (!res)
|
||||||
}
|
{
|
||||||
ntfs_attr_close(na);
|
/*
|
||||||
} else
|
* open and update the existing efs data
|
||||||
res = -1;
|
*/
|
||||||
}
|
na = ntfs_attr_open(ni, AT_LOGGED_UTILITY_STREAM,
|
||||||
if (!res) {
|
logged_utility_stream_name, 4);
|
||||||
/* Don't handle AT_DATA Attribute(s) if inode is a directory */
|
if (na)
|
||||||
if (!(ni->mrec->flags & MFT_RECORD_IS_DIRECTORY)) {
|
{
|
||||||
/* iterate over AT_DATA attributes */
|
/* resize attribute */
|
||||||
/* set encrypted flag, truncate attribute to match padding bytes */
|
res = ntfs_attr_truncate(na, (s64)size);
|
||||||
|
/* overwrite value if any */
|
||||||
if (fixup_loop(ni))
|
if (!res && value)
|
||||||
return -1;
|
{
|
||||||
}
|
written = (int)ntfs_attr_pwrite(na,
|
||||||
ni->flags |= FILE_ATTR_ENCRYPTED;
|
(s64)0, (s64)size, value);
|
||||||
NInoSetDirty(ni);
|
if (written != (s64)size)
|
||||||
NInoFileNameSetDirty(ni);
|
{
|
||||||
}
|
ntfs_log_error("Failed to "
|
||||||
} else {
|
"update efs data\n");
|
||||||
errno = EINVAL;
|
errno = EIO;
|
||||||
res = -1;
|
res = -1;
|
||||||
}
|
}
|
||||||
return (res ? -1 : 0);
|
}
|
||||||
|
ntfs_attr_close(na);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
res = -1;
|
||||||
|
}
|
||||||
|
if (!res)
|
||||||
|
{
|
||||||
|
/* Don't handle AT_DATA Attribute(s) if inode is a directory */
|
||||||
|
if (!(ni->mrec->flags & MFT_RECORD_IS_DIRECTORY))
|
||||||
|
{
|
||||||
|
/* iterate over AT_DATA attributes */
|
||||||
|
/* set encrypted flag, truncate attribute to match padding bytes */
|
||||||
|
|
||||||
|
if (fixup_loop(ni))
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
ni->flags |= FILE_ATTR_ENCRYPTED;
|
||||||
|
NInoSetDirty(ni);
|
||||||
|
NInoFileNameSetDirty(ni);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
errno = EINVAL;
|
||||||
|
res = -1;
|
||||||
|
}
|
||||||
|
return (res ? -1 : 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -321,119 +369,138 @@ int ntfs_set_efs_info(ntfs_inode *ni, const char *value, size_t size,
|
|||||||
* -1 if failed (errno tells why)
|
* -1 if failed (errno tells why)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
int ntfs_efs_fixup_attribute(ntfs_attr_search_ctx *ctx, ntfs_attr *na)
|
int ntfs_efs_fixup_attribute(ntfs_attr_search_ctx *ctx, ntfs_attr *na)
|
||||||
{
|
{
|
||||||
u64 newsize;
|
u64 newsize;
|
||||||
u64 oldsize;
|
u64 oldsize;
|
||||||
le16 appended_bytes;
|
le16 appended_bytes;
|
||||||
u16 padding_length;
|
u16 padding_length;
|
||||||
ntfs_inode *ni;
|
ntfs_inode *ni;
|
||||||
BOOL close_ctx = FALSE;
|
BOOL close_ctx = FALSE;
|
||||||
|
|
||||||
if (!na) {
|
if (!na)
|
||||||
ntfs_log_error("no na specified for efs_fixup_attribute\n");
|
{
|
||||||
goto err_out;
|
ntfs_log_error("no na specified for efs_fixup_attribute\n");
|
||||||
}
|
goto err_out;
|
||||||
if (!ctx) {
|
}
|
||||||
ctx = ntfs_attr_get_search_ctx(na->ni, NULL);
|
if (!ctx)
|
||||||
if (!ctx) {
|
{
|
||||||
ntfs_log_error("Failed to get ctx for efs\n");
|
ctx = ntfs_attr_get_search_ctx(na->ni, NULL);
|
||||||
goto err_out;
|
if (!ctx)
|
||||||
}
|
{
|
||||||
close_ctx = TRUE;
|
ntfs_log_error("Failed to get ctx for efs\n");
|
||||||
if (ntfs_attr_lookup(AT_DATA, na->name, na->name_len,
|
goto err_out;
|
||||||
CASE_SENSITIVE, 0, NULL, 0, ctx)) {
|
}
|
||||||
ntfs_log_error("attr lookup for AT_DATA attribute failed in efs fixup\n");
|
close_ctx = TRUE;
|
||||||
goto err_out;
|
if (ntfs_attr_lookup(AT_DATA, na->name, na->name_len,
|
||||||
}
|
CASE_SENSITIVE, 0, NULL, 0, ctx))
|
||||||
} else {
|
{
|
||||||
if (!NAttrNonResident(na)) {
|
ntfs_log_error("attr lookup for AT_DATA attribute failed in efs fixup\n");
|
||||||
ntfs_log_error("Cannot make non resident"
|
goto err_out;
|
||||||
" when a context has been allocated\n");
|
}
|
||||||
goto err_out;
|
}
|
||||||
}
|
else
|
||||||
}
|
{
|
||||||
|
if (!NAttrNonResident(na))
|
||||||
|
{
|
||||||
|
ntfs_log_error("Cannot make non resident"
|
||||||
|
" when a context has been allocated\n");
|
||||||
|
goto err_out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* no extra bytes are added to void attributes */
|
/* no extra bytes are added to void attributes */
|
||||||
oldsize = na->data_size;
|
oldsize = na->data_size;
|
||||||
if (oldsize) {
|
if (oldsize)
|
||||||
/* make sure size is valid for a raw encrypted stream */
|
{
|
||||||
if ((oldsize & 511) != 2) {
|
/* make sure size is valid for a raw encrypted stream */
|
||||||
ntfs_log_error("Bad raw encrypted stream\n");
|
if ((oldsize & 511) != 2)
|
||||||
goto err_out;
|
{
|
||||||
}
|
ntfs_log_error("Bad raw encrypted stream\n");
|
||||||
/* read padding length from last two bytes of attribute */
|
goto err_out;
|
||||||
if (ntfs_attr_pread(na, oldsize - 2, 2, &appended_bytes) != 2) {
|
}
|
||||||
ntfs_log_error("Error reading padding length\n");
|
/* read padding length from last two bytes of attribute */
|
||||||
goto err_out;
|
if (ntfs_attr_pread(na, oldsize - 2, 2, &appended_bytes) != 2)
|
||||||
}
|
{
|
||||||
padding_length = le16_to_cpu(appended_bytes);
|
ntfs_log_error("Error reading padding length\n");
|
||||||
if (padding_length > 511 || padding_length > na->data_size-2) {
|
goto err_out;
|
||||||
errno = EINVAL;
|
}
|
||||||
ntfs_log_error("invalid padding length %d for data_size %lld\n",
|
padding_length = le16_to_cpu(appended_bytes);
|
||||||
padding_length, (long long)oldsize);
|
if (padding_length > 511 || padding_length > na->data_size-2)
|
||||||
goto err_out;
|
{
|
||||||
}
|
errno = EINVAL;
|
||||||
newsize = oldsize - padding_length - 2;
|
ntfs_log_error("invalid padding length %d for data_size %lld\n",
|
||||||
/*
|
padding_length, (long long)oldsize);
|
||||||
* truncate attribute to possibly free clusters allocated
|
goto err_out;
|
||||||
* for the last two bytes, but do not truncate to new size
|
}
|
||||||
* to avoid losing useful data
|
newsize = oldsize - padding_length - 2;
|
||||||
*/
|
/*
|
||||||
if (ntfs_attr_truncate(na, oldsize - 2)) {
|
* truncate attribute to possibly free clusters allocated
|
||||||
ntfs_log_error("Error truncating attribute\n");
|
* for the last two bytes, but do not truncate to new size
|
||||||
goto err_out;
|
* to avoid losing useful data
|
||||||
}
|
*/
|
||||||
} else
|
if (ntfs_attr_truncate(na, oldsize - 2))
|
||||||
newsize = 0;
|
{
|
||||||
|
ntfs_log_error("Error truncating attribute\n");
|
||||||
|
goto err_out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
newsize = 0;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Encrypted AT_DATA Attributes MUST be non-resident
|
* Encrypted AT_DATA Attributes MUST be non-resident
|
||||||
* This has to be done after the attribute is resized, as
|
* This has to be done after the attribute is resized, as
|
||||||
* resizing down to zero may cause the attribute to be made
|
* resizing down to zero may cause the attribute to be made
|
||||||
* resident.
|
* resident.
|
||||||
*/
|
*/
|
||||||
if (!NAttrNonResident(na)
|
if (!NAttrNonResident(na)
|
||||||
&& ntfs_attr_make_non_resident(na, ctx)) {
|
&& ntfs_attr_make_non_resident(na, ctx))
|
||||||
if (!close_ctx
|
{
|
||||||
|| ntfs_attr_force_non_resident(na)) {
|
if (!close_ctx
|
||||||
ntfs_log_error("Error making DATA attribute non-resident\n");
|
|| ntfs_attr_force_non_resident(na))
|
||||||
goto err_out;
|
{
|
||||||
} else {
|
ntfs_log_error("Error making DATA attribute non-resident\n");
|
||||||
/*
|
goto err_out;
|
||||||
* must reinitialize context after forcing
|
}
|
||||||
* non-resident. We need a context for updating
|
else
|
||||||
* the state, and at this point, we are sure
|
{
|
||||||
* the context is not used elsewhere.
|
/*
|
||||||
*/
|
* must reinitialize context after forcing
|
||||||
ntfs_attr_reinit_search_ctx(ctx);
|
* non-resident. We need a context for updating
|
||||||
if (ntfs_attr_lookup(AT_DATA, na->name, na->name_len,
|
* the state, and at this point, we are sure
|
||||||
CASE_SENSITIVE, 0, NULL, 0, ctx)) {
|
* the context is not used elsewhere.
|
||||||
ntfs_log_error("attr lookup for AT_DATA attribute failed in efs fixup\n");
|
*/
|
||||||
goto err_out;
|
ntfs_attr_reinit_search_ctx(ctx);
|
||||||
}
|
if (ntfs_attr_lookup(AT_DATA, na->name, na->name_len,
|
||||||
}
|
CASE_SENSITIVE, 0, NULL, 0, ctx))
|
||||||
}
|
{
|
||||||
ni = na->ni;
|
ntfs_log_error("attr lookup for AT_DATA attribute failed in efs fixup\n");
|
||||||
if (!na->name_len) {
|
goto err_out;
|
||||||
ni->data_size = newsize;
|
}
|
||||||
ni->allocated_size = na->allocated_size;
|
}
|
||||||
}
|
}
|
||||||
NInoSetDirty(ni);
|
ni = na->ni;
|
||||||
NInoFileNameSetDirty(ni);
|
if (!na->name_len)
|
||||||
|
{
|
||||||
|
ni->data_size = newsize;
|
||||||
|
ni->allocated_size = na->allocated_size;
|
||||||
|
}
|
||||||
|
NInoSetDirty(ni);
|
||||||
|
NInoFileNameSetDirty(ni);
|
||||||
|
|
||||||
ctx->attr->data_size = cpu_to_le64(newsize);
|
ctx->attr->data_size = cpu_to_le64(newsize);
|
||||||
if (le64_to_cpu(ctx->attr->initialized_size) > newsize)
|
if (le64_to_cpu(ctx->attr->initialized_size) > newsize)
|
||||||
ctx->attr->initialized_size = ctx->attr->data_size;
|
ctx->attr->initialized_size = ctx->attr->data_size;
|
||||||
ctx->attr->flags |= ATTR_IS_ENCRYPTED;
|
ctx->attr->flags |= ATTR_IS_ENCRYPTED;
|
||||||
if (close_ctx)
|
if (close_ctx)
|
||||||
ntfs_attr_put_search_ctx(ctx);
|
ntfs_attr_put_search_ctx(ctx);
|
||||||
|
|
||||||
return (0);
|
return (0);
|
||||||
err_out:
|
err_out:
|
||||||
if (close_ctx && ctx)
|
if (close_ctx && ctx)
|
||||||
ntfs_attr_put_search_ctx(ctx);
|
ntfs_attr_put_search_ctx(ctx);
|
||||||
return (-1);
|
return (-1);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif /* HAVE_SETXATTR */
|
#endif /* HAVE_SETXATTR */
|
||||||
|
@ -23,8 +23,7 @@
|
|||||||
|
|
||||||
int ntfs_get_efs_info(ntfs_inode *ni, char *value, size_t size);
|
int ntfs_get_efs_info(ntfs_inode *ni, char *value, size_t size);
|
||||||
|
|
||||||
int ntfs_set_efs_info(ntfs_inode *ni,
|
int ntfs_set_efs_info(ntfs_inode *ni, const char *value, size_t size, int flags);
|
||||||
const char *value, size_t size, int flags);
|
|
||||||
int ntfs_efs_fixup_attribute(ntfs_attr_search_ctx *ctx, ntfs_attr *na);
|
int ntfs_efs_fixup_attribute(ntfs_attr_search_ctx *ctx, ntfs_attr *na);
|
||||||
|
|
||||||
#endif /* EFS_H */
|
#endif /* EFS_H */
|
||||||
|
@ -69,7 +69,8 @@
|
|||||||
static s64 ntfs_device_gekko_io_readbytes(struct ntfs_device *dev, s64 offset, s64 count, void *buf);
|
static s64 ntfs_device_gekko_io_readbytes(struct ntfs_device *dev, s64 offset, s64 count, void *buf);
|
||||||
static bool ntfs_device_gekko_io_readsectors(struct ntfs_device *dev, sec_t sector, sec_t numSectors, void* buffer);
|
static bool ntfs_device_gekko_io_readsectors(struct ntfs_device *dev, sec_t sector, sec_t numSectors, void* buffer);
|
||||||
static s64 ntfs_device_gekko_io_writebytes(struct ntfs_device *dev, s64 offset, s64 count, const void *buf);
|
static s64 ntfs_device_gekko_io_writebytes(struct ntfs_device *dev, s64 offset, s64 count, const void *buf);
|
||||||
static bool ntfs_device_gekko_io_writesectors(struct ntfs_device *dev, sec_t sector, sec_t numSectors, const void* buffer);
|
static bool ntfs_device_gekko_io_writesectors(struct ntfs_device *dev, sec_t sector, sec_t numSectors,
|
||||||
|
const void* buffer);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
@ -80,32 +81,37 @@ static int ntfs_device_gekko_io_open(struct ntfs_device *dev, int flags)
|
|||||||
|
|
||||||
// Get the device driver descriptor
|
// Get the device driver descriptor
|
||||||
gekko_fd *fd = DEV_FD(dev);
|
gekko_fd *fd = DEV_FD(dev);
|
||||||
if (!fd) {
|
if (!fd)
|
||||||
|
{
|
||||||
errno = EBADF;
|
errno = EBADF;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the device interface
|
// Get the device interface
|
||||||
const DISC_INTERFACE* interface = fd->interface;
|
const DISC_INTERFACE* interface = fd->interface;
|
||||||
if (!interface) {
|
if (!interface)
|
||||||
|
{
|
||||||
errno = ENODEV;
|
errno = ENODEV;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start the device interface and ensure that it is inserted
|
// Start the device interface and ensure that it is inserted
|
||||||
if (!interface->startup()) {
|
if (!interface->startup())
|
||||||
|
{
|
||||||
ntfs_log_perror("device failed to start\n");
|
ntfs_log_perror("device failed to start\n");
|
||||||
errno = EIO;
|
errno = EIO;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
if (!interface->isInserted()) {
|
if (!interface->isInserted())
|
||||||
|
{
|
||||||
ntfs_log_perror("device media is not inserted\n");
|
ntfs_log_perror("device media is not inserted\n");
|
||||||
errno = EIO;
|
errno = EIO;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check that the device isn't already open (used by another volume?)
|
// Check that the device isn't already open (used by another volume?)
|
||||||
if (NDevOpen(dev)) {
|
if (NDevOpen(dev))
|
||||||
|
{
|
||||||
ntfs_log_perror("device is busy (already open)\n");
|
ntfs_log_perror("device is busy (already open)\n");
|
||||||
errno = EBUSY;
|
errno = EBUSY;
|
||||||
return -1;
|
return -1;
|
||||||
@ -113,12 +119,16 @@ static int ntfs_device_gekko_io_open(struct ntfs_device *dev, int flags)
|
|||||||
|
|
||||||
// Check that there is a valid NTFS boot sector at the start of the device
|
// Check that there is a valid NTFS boot sector at the start of the device
|
||||||
NTFS_BOOT_SECTOR boot;
|
NTFS_BOOT_SECTOR boot;
|
||||||
if (interface->readSectors(fd->startSector, 1, &boot)) {
|
if (interface->readSectors(fd->startSector, 1, &boot))
|
||||||
if (!ntfs_boot_sector_is_ntfs(&boot)) {
|
{
|
||||||
|
if (!ntfs_boot_sector_is_ntfs(&boot))
|
||||||
|
{
|
||||||
errno = EINVALPART;
|
errno = EINVALPART;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
ntfs_log_perror("read failure @ sector %d\n", fd->startSector);
|
ntfs_log_perror("read failure @ sector %d\n", fd->startSector);
|
||||||
errno = EIO;
|
errno = EIO;
|
||||||
return -1;
|
return -1;
|
||||||
@ -133,12 +143,14 @@ static int ntfs_device_gekko_io_open(struct ntfs_device *dev, int flags)
|
|||||||
fd->ino = le64_to_cpu(boot.volume_serial_number);
|
fd->ino = le64_to_cpu(boot.volume_serial_number);
|
||||||
|
|
||||||
// Mark the device as read-only (if required)
|
// Mark the device as read-only (if required)
|
||||||
if (flags & O_RDONLY) {
|
if (flags & O_RDONLY)
|
||||||
|
{
|
||||||
NDevSetReadOnly(dev);
|
NDevSetReadOnly(dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create the cache
|
// Create the cache
|
||||||
fd->cache = _NTFS_cache_constructor(fd->cachePageCount, fd->cachePageSize, interface, fd->startSector + fd->sectorCount, fd->sectorSize);
|
fd->cache = _NTFS_cache_constructor(fd->cachePageCount, fd->cachePageSize, interface, fd->startSector
|
||||||
|
+ fd->sectorCount, fd->sectorSize);
|
||||||
|
|
||||||
// Mark the device as open
|
// Mark the device as open
|
||||||
NDevSetBlock(dev);
|
NDevSetBlock(dev);
|
||||||
@ -156,13 +168,15 @@ static int ntfs_device_gekko_io_close(struct ntfs_device *dev)
|
|||||||
|
|
||||||
// Get the device driver descriptor
|
// Get the device driver descriptor
|
||||||
gekko_fd *fd = DEV_FD(dev);
|
gekko_fd *fd = DEV_FD(dev);
|
||||||
if (!fd) {
|
if (!fd)
|
||||||
|
{
|
||||||
errno = EBADF;
|
errno = EBADF;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check that the device is actually open
|
// Check that the device is actually open
|
||||||
if (!NDevOpen(dev)) {
|
if (!NDevOpen(dev))
|
||||||
|
{
|
||||||
ntfs_log_perror("device is not open\n");
|
ntfs_log_perror("device is not open\n");
|
||||||
errno = EIO;
|
errno = EIO;
|
||||||
return -1;
|
return -1;
|
||||||
@ -173,7 +187,8 @@ static int ntfs_device_gekko_io_close(struct ntfs_device *dev)
|
|||||||
NDevClearBlock(dev);
|
NDevClearBlock(dev);
|
||||||
|
|
||||||
// Flush the device (if dirty and not read-only)
|
// Flush the device (if dirty and not read-only)
|
||||||
if (NDevDirty(dev) && !NDevReadOnly(dev)) {
|
if (NDevDirty(dev) && !NDevReadOnly(dev))
|
||||||
|
{
|
||||||
ntfs_log_debug("device is dirty, will now sync\n");
|
ntfs_log_debug("device is dirty, will now sync\n");
|
||||||
|
|
||||||
// ...?
|
// ...?
|
||||||
@ -184,16 +199,17 @@ static int ntfs_device_gekko_io_close(struct ntfs_device *dev)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Flush and destroy the cache (if required)
|
// Flush and destroy the cache (if required)
|
||||||
if (fd->cache) {
|
if (fd->cache)
|
||||||
|
{
|
||||||
_NTFS_cache_flush(fd->cache);
|
_NTFS_cache_flush(fd->cache);
|
||||||
_NTFS_cache_destructor(fd->cache);
|
_NTFS_cache_destructor(fd->cache);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Shutdown the device interface
|
// Shutdown the device interface
|
||||||
/*const DISC_INTERFACE* interface = fd->interface;
|
/*const DISC_INTERFACE* interface = fd->interface;
|
||||||
if (interface) {
|
if (interface) {
|
||||||
interface->shutdown();
|
interface->shutdown();
|
||||||
}*/
|
}*/
|
||||||
|
|
||||||
// Free the device driver private data
|
// Free the device driver private data
|
||||||
ntfs_free(dev->d_private);
|
ntfs_free(dev->d_private);
|
||||||
@ -211,16 +227,24 @@ static s64 ntfs_device_gekko_io_seek(struct ntfs_device *dev, s64 offset, int wh
|
|||||||
|
|
||||||
// Get the device driver descriptor
|
// Get the device driver descriptor
|
||||||
gekko_fd *fd = DEV_FD(dev);
|
gekko_fd *fd = DEV_FD(dev);
|
||||||
if (!fd) {
|
if (!fd)
|
||||||
|
{
|
||||||
errno = EBADF;
|
errno = EBADF;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set the current position on the device (in bytes)
|
// Set the current position on the device (in bytes)
|
||||||
switch(whence) {
|
switch (whence)
|
||||||
case SEEK_SET: fd->pos = MIN(MAX(offset, 0), fd->len); break;
|
{
|
||||||
case SEEK_CUR: fd->pos = MIN(MAX(fd->pos + offset, 0), fd->len); break;
|
case SEEK_SET:
|
||||||
case SEEK_END: fd->pos = MIN(MAX(fd->len + offset, 0), fd->len); break;
|
fd->pos = MIN(MAX(offset, 0), fd->len);
|
||||||
|
break;
|
||||||
|
case SEEK_CUR:
|
||||||
|
fd->pos = MIN(MAX(fd->pos + offset, 0), fd->len);
|
||||||
|
break;
|
||||||
|
case SEEK_END:
|
||||||
|
fd->pos = MIN(MAX(fd->len + offset, 0), fd->len);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -267,26 +291,27 @@ static s64 ntfs_device_gekko_io_readbytes(struct ntfs_device *dev, s64 offset, s
|
|||||||
|
|
||||||
// Get the device driver descriptor
|
// Get the device driver descriptor
|
||||||
gekko_fd *fd = DEV_FD(dev);
|
gekko_fd *fd = DEV_FD(dev);
|
||||||
if (!fd) {
|
if (!fd)
|
||||||
|
{
|
||||||
errno = EBADF;
|
errno = EBADF;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the device interface
|
// Get the device interface
|
||||||
const DISC_INTERFACE* interface = fd->interface;
|
const DISC_INTERFACE* interface = fd->interface;
|
||||||
if (!interface) {
|
if (!interface)
|
||||||
|
{
|
||||||
errno = ENODEV;
|
errno = ENODEV;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(offset < 0)
|
if (offset < 0)
|
||||||
{
|
{
|
||||||
errno = EROFS;
|
errno = EROFS;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!count)
|
if (!count) return 0;
|
||||||
return 0;
|
|
||||||
|
|
||||||
sec_t sec_start = (sec_t) fd->startSector;
|
sec_t sec_start = (sec_t) fd->startSector;
|
||||||
sec_t sec_count = 1;
|
sec_t sec_count = 1;
|
||||||
@ -294,33 +319,38 @@ static s64 ntfs_device_gekko_io_readbytes(struct ntfs_device *dev, s64 offset, s
|
|||||||
u8 *buffer = NULL;
|
u8 *buffer = NULL;
|
||||||
|
|
||||||
// Determine the range of sectors required for this read
|
// Determine the range of sectors required for this read
|
||||||
if (offset > 0) {
|
if (offset > 0)
|
||||||
|
{
|
||||||
sec_start += (sec_t) floor((f64) offset / (f64) fd->sectorSize);
|
sec_start += (sec_t) floor((f64) offset / (f64) fd->sectorSize);
|
||||||
}
|
}
|
||||||
if (buffer_offset+count > fd->sectorSize) {
|
if (buffer_offset + count > fd->sectorSize)
|
||||||
sec_count = (sec_t) ceil((f64) (buffer_offset+count) / (f64) fd->sectorSize);
|
{
|
||||||
|
sec_count = (sec_t) ceil((f64) (buffer_offset + count) / (f64) fd->sectorSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
// If this read happens to be on the sector boundaries then do the read straight into the destination buffer
|
// If this read happens to be on the sector boundaries then do the read straight into the destination buffer
|
||||||
|
|
||||||
if((buffer_offset == 0) && (count % fd->sectorSize == 0)) {
|
if ((buffer_offset == 0) && (count % fd->sectorSize == 0))
|
||||||
|
{
|
||||||
|
|
||||||
// Read from the device
|
// Read from the device
|
||||||
ntfs_log_trace("direct read from sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
ntfs_log_trace("direct read from sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
||||||
if (!ntfs_device_gekko_io_readsectors(dev, sec_start, sec_count, buf)) {
|
if (!ntfs_device_gekko_io_readsectors(dev, sec_start, sec_count, buf))
|
||||||
|
{
|
||||||
ntfs_log_perror("direct read failure @ sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
ntfs_log_perror("direct read failure @ sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
||||||
errno = EIO;
|
errno = EIO;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Else read into a buffer and copy over only what was requested
|
// Else read into a buffer and copy over only what was requested
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
|
||||||
// Allocate a buffer to hold the read data
|
// Allocate a buffer to hold the read data
|
||||||
buffer = (u8*)ntfs_alloc(sec_count * fd->sectorSize);
|
buffer = (u8*) ntfs_alloc(sec_count * fd->sectorSize);
|
||||||
if (!buffer) {
|
if (!buffer)
|
||||||
|
{
|
||||||
errno = ENOMEM;
|
errno = ENOMEM;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@ -328,7 +358,8 @@ static s64 ntfs_device_gekko_io_readbytes(struct ntfs_device *dev, s64 offset, s
|
|||||||
// Read from the device
|
// Read from the device
|
||||||
ntfs_log_trace("buffered read from sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
ntfs_log_trace("buffered read from sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
||||||
ntfs_log_trace("count: %d sec_count:%d fd->sectorSize: %d )\n", (u32)count, (u32)sec_count,(u32)fd->sectorSize);
|
ntfs_log_trace("count: %d sec_count:%d fd->sectorSize: %d )\n", (u32)count, (u32)sec_count,(u32)fd->sectorSize);
|
||||||
if (!ntfs_device_gekko_io_readsectors(dev, sec_start, sec_count, buffer)) {
|
if (!ntfs_device_gekko_io_readsectors(dev, sec_start, sec_count, buffer))
|
||||||
|
{
|
||||||
ntfs_log_perror("buffered read failure @ sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
ntfs_log_perror("buffered read failure @ sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
||||||
ntfs_free(buffer);
|
ntfs_free(buffer);
|
||||||
errno = EIO;
|
errno = EIO;
|
||||||
@ -353,31 +384,34 @@ static s64 ntfs_device_gekko_io_writebytes(struct ntfs_device *dev, s64 offset,
|
|||||||
|
|
||||||
// Get the device driver descriptor
|
// Get the device driver descriptor
|
||||||
gekko_fd *fd = DEV_FD(dev);
|
gekko_fd *fd = DEV_FD(dev);
|
||||||
if (!fd) {
|
if (!fd)
|
||||||
|
{
|
||||||
errno = EBADF;
|
errno = EBADF;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the device interface
|
// Get the device interface
|
||||||
const DISC_INTERFACE* interface = fd->interface;
|
const DISC_INTERFACE* interface = fd->interface;
|
||||||
if (!interface) {
|
if (!interface)
|
||||||
|
{
|
||||||
errno = ENODEV;
|
errno = ENODEV;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check that the device can be written to
|
// Check that the device can be written to
|
||||||
if (NDevReadOnly(dev)) {
|
if (NDevReadOnly(dev))
|
||||||
|
{
|
||||||
errno = EROFS;
|
errno = EROFS;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(count < 0 || offset < 0) {
|
if (count < 0 || offset < 0)
|
||||||
|
{
|
||||||
errno = EROFS;
|
errno = EROFS;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(count == 0)
|
if (count == 0) return 0;
|
||||||
return 0;
|
|
||||||
|
|
||||||
sec_t sec_start = (sec_t) fd->startSector;
|
sec_t sec_start = (sec_t) fd->startSector;
|
||||||
sec_t sec_count = 1;
|
sec_t sec_count = 1;
|
||||||
@ -385,48 +419,55 @@ static s64 ntfs_device_gekko_io_writebytes(struct ntfs_device *dev, s64 offset,
|
|||||||
u8 *buffer = NULL;
|
u8 *buffer = NULL;
|
||||||
|
|
||||||
// Determine the range of sectors required for this write
|
// Determine the range of sectors required for this write
|
||||||
if (offset > 0) {
|
if (offset > 0)
|
||||||
|
{
|
||||||
sec_start += (sec_t) floor((f64) offset / (f64) fd->sectorSize);
|
sec_start += (sec_t) floor((f64) offset / (f64) fd->sectorSize);
|
||||||
}
|
}
|
||||||
if ((buffer_offset+count) > fd->sectorSize) {
|
if ((buffer_offset + count) > fd->sectorSize)
|
||||||
sec_count = (sec_t) ceil((f64) (buffer_offset+count) / (f64) fd->sectorSize);
|
{
|
||||||
|
sec_count = (sec_t) ceil((f64) (buffer_offset + count) / (f64) fd->sectorSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
// If this write happens to be on the sector boundaries then do the write straight to disc
|
// If this write happens to be on the sector boundaries then do the write straight to disc
|
||||||
if((buffer_offset == 0) && (count % fd->sectorSize == 0))
|
if ((buffer_offset == 0) && (count % fd->sectorSize == 0))
|
||||||
{
|
{
|
||||||
// Write to the device
|
// Write to the device
|
||||||
ntfs_log_trace("direct write to sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
ntfs_log_trace("direct write to sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
||||||
if (!ntfs_device_gekko_io_writesectors(dev, sec_start, sec_count, buf)) {
|
if (!ntfs_device_gekko_io_writesectors(dev, sec_start, sec_count, buf))
|
||||||
|
{
|
||||||
ntfs_log_perror("direct write failure @ sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
ntfs_log_perror("direct write failure @ sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
||||||
errno = EIO;
|
errno = EIO;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
// Else write from a buffer aligned to the sector boundaries
|
// Else write from a buffer aligned to the sector boundaries
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Allocate a buffer to hold the write data
|
// Allocate a buffer to hold the write data
|
||||||
buffer = (u8 *) ntfs_alloc(sec_count * fd->sectorSize);
|
buffer = (u8 *) ntfs_alloc(sec_count * fd->sectorSize);
|
||||||
if (!buffer) {
|
if (!buffer)
|
||||||
|
{
|
||||||
errno = ENOMEM;
|
errno = ENOMEM;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
// Read the first and last sectors of the buffer from disc (if required)
|
// Read the first and last sectors of the buffer from disc (if required)
|
||||||
// NOTE: This is done because the data does not line up with the sector boundaries,
|
// NOTE: This is done because the data does not line up with the sector boundaries,
|
||||||
// we just read in the buffer edges where the data overlaps with the rest of the disc
|
// we just read in the buffer edges where the data overlaps with the rest of the disc
|
||||||
if(buffer_offset != 0)
|
if (buffer_offset != 0)
|
||||||
{
|
{
|
||||||
if (!ntfs_device_gekko_io_readsectors(dev, sec_start, 1, buffer)) {
|
if (!ntfs_device_gekko_io_readsectors(dev, sec_start, 1, buffer))
|
||||||
|
{
|
||||||
ntfs_log_perror("read failure @ sector %d\n", sec_start);
|
ntfs_log_perror("read failure @ sector %d\n", sec_start);
|
||||||
ntfs_free(buffer);
|
ntfs_free(buffer);
|
||||||
errno = EIO;
|
errno = EIO;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if((buffer_offset+count) % fd->sectorSize != 0)
|
if ((buffer_offset + count) % fd->sectorSize != 0)
|
||||||
{
|
{
|
||||||
if (!ntfs_device_gekko_io_readsectors(dev, sec_start + sec_count - 1, 1, buffer + ((sec_count-1) * fd->sectorSize))) {
|
if (!ntfs_device_gekko_io_readsectors(dev, sec_start + sec_count - 1, 1, buffer + ((sec_count - 1)
|
||||||
|
* fd->sectorSize)))
|
||||||
|
{
|
||||||
ntfs_log_perror("read failure @ sector %d\n", sec_start + sec_count - 1);
|
ntfs_log_perror("read failure @ sector %d\n", sec_start + sec_count - 1);
|
||||||
ntfs_free(buffer);
|
ntfs_free(buffer);
|
||||||
errno = EIO;
|
errno = EIO;
|
||||||
@ -439,7 +480,8 @@ static s64 ntfs_device_gekko_io_writebytes(struct ntfs_device *dev, s64 offset,
|
|||||||
|
|
||||||
// Write to the device
|
// Write to the device
|
||||||
ntfs_log_trace("buffered write to sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
ntfs_log_trace("buffered write to sector %d (%d sector(s) long)\n", sec_start, sec_count);
|
||||||
if (!ntfs_device_gekko_io_writesectors(dev, sec_start, sec_count, buffer)) {
|
if (!ntfs_device_gekko_io_writesectors(dev, sec_start, sec_count, buffer))
|
||||||
|
{
|
||||||
ntfs_log_perror("buffered write failure @ sector %d\n", sec_start);
|
ntfs_log_perror("buffered write failure @ sector %d\n", sec_start);
|
||||||
ntfs_free(buffer);
|
ntfs_free(buffer);
|
||||||
errno = EIO;
|
errno = EIO;
|
||||||
@ -460,24 +502,26 @@ static bool ntfs_device_gekko_io_readsectors(struct ntfs_device *dev, sec_t sect
|
|||||||
{
|
{
|
||||||
// Get the device driver descriptor
|
// Get the device driver descriptor
|
||||||
gekko_fd *fd = DEV_FD(dev);
|
gekko_fd *fd = DEV_FD(dev);
|
||||||
if (!fd) {
|
if (!fd)
|
||||||
|
{
|
||||||
errno = EBADF;
|
errno = EBADF;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
// Read the sectors from disc (or cache, if enabled)
|
// Read the sectors from disc (or cache, if enabled)
|
||||||
if (fd->cache)
|
if (fd->cache)
|
||||||
return _NTFS_cache_readSectors(fd->cache, sector, numSectors, buffer);
|
return _NTFS_cache_readSectors(fd->cache, sector, numSectors, buffer);
|
||||||
else
|
else return fd->interface->readSectors(sector, numSectors, buffer);
|
||||||
return fd->interface->readSectors(sector, numSectors, buffer);
|
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool ntfs_device_gekko_io_writesectors(struct ntfs_device *dev, sec_t sector, sec_t numSectors, const void* buffer)
|
static bool ntfs_device_gekko_io_writesectors(struct ntfs_device *dev, sec_t sector, sec_t numSectors,
|
||||||
|
const void* buffer)
|
||||||
{
|
{
|
||||||
// Get the device driver descriptor
|
// Get the device driver descriptor
|
||||||
gekko_fd *fd = DEV_FD(dev);
|
gekko_fd *fd = DEV_FD(dev);
|
||||||
if (!fd) {
|
if (!fd)
|
||||||
|
{
|
||||||
errno = EBADF;
|
errno = EBADF;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -485,8 +529,7 @@ static bool ntfs_device_gekko_io_writesectors(struct ntfs_device *dev, sec_t sec
|
|||||||
// Write the sectors to disc (or cache, if enabled)
|
// Write the sectors to disc (or cache, if enabled)
|
||||||
if (fd->cache)
|
if (fd->cache)
|
||||||
return _NTFS_cache_writeSectors(fd->cache, sector, numSectors, buffer);
|
return _NTFS_cache_writeSectors(fd->cache, sector, numSectors, buffer);
|
||||||
else
|
else return fd->interface->writeSectors(sector, numSectors, buffer);
|
||||||
return fd->interface->writeSectors(sector, numSectors, buffer);
|
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -496,11 +539,12 @@ static bool ntfs_device_gekko_io_writesectors(struct ntfs_device *dev, sec_t sec
|
|||||||
*/
|
*/
|
||||||
static int ntfs_device_gekko_io_sync(struct ntfs_device *dev)
|
static int ntfs_device_gekko_io_sync(struct ntfs_device *dev)
|
||||||
{
|
{
|
||||||
gekko_fd *fd = DEV_FD(dev);
|
gekko_fd *fd = DEV_FD(dev);
|
||||||
ntfs_log_trace("dev %p\n", dev);
|
ntfs_log_trace("dev %p\n", dev);
|
||||||
|
|
||||||
// Check that the device can be written to
|
// Check that the device can be written to
|
||||||
if (NDevReadOnly(dev)) {
|
if (NDevReadOnly(dev))
|
||||||
|
{
|
||||||
errno = EROFS;
|
errno = EROFS;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@ -509,8 +553,10 @@ static int ntfs_device_gekko_io_sync(struct ntfs_device *dev)
|
|||||||
NDevClearDirty(dev);
|
NDevClearDirty(dev);
|
||||||
|
|
||||||
// Flush any sectors in the disc cache (if required)
|
// Flush any sectors in the disc cache (if required)
|
||||||
if (fd->cache) {
|
if (fd->cache)
|
||||||
if (!_NTFS_cache_flush(fd->cache)) {
|
{
|
||||||
|
if (!_NTFS_cache_flush(fd->cache))
|
||||||
|
{
|
||||||
errno = EIO;
|
errno = EIO;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@ -528,19 +574,18 @@ static int ntfs_device_gekko_io_stat(struct ntfs_device *dev, struct stat *buf)
|
|||||||
|
|
||||||
// Get the device driver descriptor
|
// Get the device driver descriptor
|
||||||
gekko_fd *fd = DEV_FD(dev);
|
gekko_fd *fd = DEV_FD(dev);
|
||||||
if (!fd) {
|
if (!fd)
|
||||||
|
{
|
||||||
errno = EBADF;
|
errno = EBADF;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Short circuit cases were we don't actually have to do anything
|
// Short circuit cases were we don't actually have to do anything
|
||||||
if (!buf)
|
if (!buf) return 0;
|
||||||
return 0;
|
|
||||||
|
|
||||||
// Build the device mode
|
// Build the device mode
|
||||||
mode_t mode = (S_IFBLK) |
|
mode_t mode = (S_IFBLK) | (S_IRUSR | S_IRGRP | S_IROTH)
|
||||||
(S_IRUSR | S_IRGRP | S_IROTH) |
|
| ((!NDevReadOnly(dev)) ? (S_IWUSR | S_IWGRP | S_IWOTH) : 0);
|
||||||
((!NDevReadOnly(dev)) ? (S_IWUSR | S_IWGRP | S_IWOTH) : 0);
|
|
||||||
|
|
||||||
// Zero out the stat buffer
|
// Zero out the stat buffer
|
||||||
memset(buf, 0, sizeof(struct stat));
|
memset(buf, 0, sizeof(struct stat));
|
||||||
@ -565,33 +610,38 @@ static int ntfs_device_gekko_io_ioctl(struct ntfs_device *dev, int request, void
|
|||||||
|
|
||||||
// Get the device driver descriptor
|
// Get the device driver descriptor
|
||||||
gekko_fd *fd = DEV_FD(dev);
|
gekko_fd *fd = DEV_FD(dev);
|
||||||
if (!fd) {
|
if (!fd)
|
||||||
|
{
|
||||||
errno = EBADF;
|
errno = EBADF;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Figure out which i/o control was requested
|
// Figure out which i/o control was requested
|
||||||
switch (request) {
|
switch (request)
|
||||||
|
{
|
||||||
|
|
||||||
// Get block device size (sectors)
|
// Get block device size (sectors)
|
||||||
#if defined(BLKGETSIZE)
|
#if defined(BLKGETSIZE)
|
||||||
case BLKGETSIZE: {
|
case BLKGETSIZE:
|
||||||
|
{
|
||||||
*(u32*)argp = fd->sectorCount;
|
*(u32*)argp = fd->sectorCount;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Get block device size (bytes)
|
// Get block device size (bytes)
|
||||||
#if defined(BLKGETSIZE64)
|
#if defined(BLKGETSIZE64)
|
||||||
case BLKGETSIZE64: {
|
case BLKGETSIZE64:
|
||||||
|
{
|
||||||
*(u64*)argp = (fd->sectorCount * fd->sectorSize);
|
*(u64*)argp = (fd->sectorCount * fd->sectorSize);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Get hard drive geometry
|
// Get hard drive geometry
|
||||||
#if defined(HDIO_GETGEO)
|
#if defined(HDIO_GETGEO)
|
||||||
case HDIO_GETGEO: {
|
case HDIO_GETGEO:
|
||||||
|
{
|
||||||
struct hd_geometry *geo = (struct hd_geometry*)argp;
|
struct hd_geometry *geo = (struct hd_geometry*)argp;
|
||||||
geo->sectors = 0;
|
geo->sectors = 0;
|
||||||
geo->heads = 0;
|
geo->heads = 0;
|
||||||
@ -599,27 +649,30 @@ static int ntfs_device_gekko_io_ioctl(struct ntfs_device *dev, int request, void
|
|||||||
geo->start = fd->hiddenSectors;
|
geo->start = fd->hiddenSectors;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Get block device sector size (bytes)
|
// Get block device sector size (bytes)
|
||||||
#if defined(BLKSSZGET)
|
#if defined(BLKSSZGET)
|
||||||
case BLKSSZGET: {
|
case BLKSSZGET:
|
||||||
|
{
|
||||||
*(int*)argp = fd->sectorSize;
|
*(int*)argp = fd->sectorSize;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Set block device block size (bytes)
|
// Set block device block size (bytes)
|
||||||
#if defined(BLKBSZSET)
|
#if defined(BLKBSZSET)
|
||||||
case BLKBSZSET: {
|
case BLKBSZSET:
|
||||||
|
{
|
||||||
int sectorSize = *(int*)argp;
|
int sectorSize = *(int*)argp;
|
||||||
fd->sectorSize = sectorSize;
|
fd->sectorSize = sectorSize;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Unimplemented ioctrl
|
// Unimplemented ioctrl
|
||||||
default: {
|
default:
|
||||||
|
{
|
||||||
ntfs_log_perror("Unimplemented ioctrl %i\n", request);
|
ntfs_log_perror("Unimplemented ioctrl %i\n", request);
|
||||||
errno = EOPNOTSUPP;
|
errno = EOPNOTSUPP;
|
||||||
return -1;
|
return -1;
|
||||||
@ -633,15 +686,8 @@ static int ntfs_device_gekko_io_ioctl(struct ntfs_device *dev, int request, void
|
|||||||
/**
|
/**
|
||||||
* Device operations for working with gekko style devices and files.
|
* Device operations for working with gekko style devices and files.
|
||||||
*/
|
*/
|
||||||
struct ntfs_device_operations ntfs_device_gekko_io_ops = {
|
struct ntfs_device_operations ntfs_device_gekko_io_ops = { .open = ntfs_device_gekko_io_open,
|
||||||
.open = ntfs_device_gekko_io_open,
|
.close = ntfs_device_gekko_io_close, .seek = ntfs_device_gekko_io_seek, .read = ntfs_device_gekko_io_read,
|
||||||
.close = ntfs_device_gekko_io_close,
|
.write = ntfs_device_gekko_io_write, .pread = ntfs_device_gekko_io_pread,
|
||||||
.seek = ntfs_device_gekko_io_seek,
|
.pwrite = ntfs_device_gekko_io_pwrite, .sync = ntfs_device_gekko_io_sync, .stat = ntfs_device_gekko_io_stat,
|
||||||
.read = ntfs_device_gekko_io_read,
|
.ioctl = ntfs_device_gekko_io_ioctl, };
|
||||||
.write = ntfs_device_gekko_io_write,
|
|
||||||
.pread = ntfs_device_gekko_io_pread,
|
|
||||||
.pwrite = ntfs_device_gekko_io_pwrite,
|
|
||||||
.sync = ntfs_device_gekko_io_sync,
|
|
||||||
.stat = ntfs_device_gekko_io_stat,
|
|
||||||
.ioctl = ntfs_device_gekko_io_ioctl,
|
|
||||||
};
|
|
||||||
|
@ -1,22 +1,22 @@
|
|||||||
/*
|
/*
|
||||||
* gekko_io.h - Platform specifics for device io.
|
* gekko_io.h - Platform specifics for device io.
|
||||||
*
|
*
|
||||||
* Copyright (c) 2009 Rhys "Shareese" Koedijk
|
* Copyright (c) 2009 Rhys "Shareese" Koedijk
|
||||||
*
|
*
|
||||||
* This program/include file is free software; you can redistribute it and/or
|
* This program/include file is free software; you can redistribute it and/or
|
||||||
* modify it under the terms of the GNU General Public License as published
|
* modify it under the terms of the GNU General Public License as published
|
||||||
* by the Free Software Foundation; either version 2 of the License, or
|
* by the Free Software Foundation; either version 2 of the License, or
|
||||||
* (at your option) any later version.
|
* (at your option) any later version.
|
||||||
*
|
*
|
||||||
* This program/include file is distributed in the hope that it will be
|
* This program/include file is distributed in the hope that it will be
|
||||||
* useful, but WITHOUT ANY WARRANTY; without even the implied warranty
|
* useful, but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||||
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU General Public License for more details.
|
* GNU General Public License for more details.
|
||||||
*
|
*
|
||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program; if not, write to the Free Software Foundation,
|
* along with this program; if not, write to the Free Software Foundation,
|
||||||
* Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
* Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef _GEKKO_IO_H
|
#ifndef _GEKKO_IO_H
|
||||||
#define _GEKKO_IO_H
|
#define _GEKKO_IO_H
|
||||||
@ -33,18 +33,19 @@
|
|||||||
/**
|
/**
|
||||||
* gekko_fd - Gekko device driver descriptor
|
* gekko_fd - Gekko device driver descriptor
|
||||||
*/
|
*/
|
||||||
typedef struct _gekko_fd {
|
typedef struct _gekko_fd
|
||||||
const DISC_INTERFACE* interface; /* Device disc interface */
|
{
|
||||||
sec_t startSector; /* LBA of partition start */
|
const DISC_INTERFACE* interface; /* Device disc interface */
|
||||||
sec_t hiddenSectors; /* LBA offset to true partition start (as described by boot sector) */
|
sec_t startSector; /* LBA of partition start */
|
||||||
u16 sectorSize; /* Device sector size (in bytes) */
|
sec_t hiddenSectors; /* LBA offset to true partition start (as described by boot sector) */
|
||||||
u64 sectorCount; /* Total number of sectors in partition */
|
u16 sectorSize; /* Device sector size (in bytes) */
|
||||||
u64 pos; /* Current position within the partition (in bytes) */
|
u64 sectorCount; /* Total number of sectors in partition */
|
||||||
u64 len; /* Total length of partition (in bytes) */
|
u64 pos; /* Current position within the partition (in bytes) */
|
||||||
ino_t ino; /* Device identifier */
|
u64 len; /* Total length of partition (in bytes) */
|
||||||
NTFS_CACHE *cache; /* Cache */
|
ino_t ino; /* Device identifier */
|
||||||
u32 cachePageCount; /* The number of pages in the cache */
|
NTFS_CACHE *cache; /* Cache */
|
||||||
u32 cachePageSize; /* The number of sectors per cache page */
|
u32 cachePageCount; /* The number of pages in the cache */
|
||||||
|
u32 cachePageSize; /* The number of sectors per cache page */
|
||||||
} gekko_fd;
|
} gekko_fd;
|
||||||
|
|
||||||
/* Forward declarations */
|
/* Forward declarations */
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -63,8 +63,7 @@
|
|||||||
|
|
||||||
#define MAX_PARENT_VCN 32
|
#define MAX_PARENT_VCN 32
|
||||||
|
|
||||||
typedef int (*COLLATE)(ntfs_volume *vol, const void *data1, int len1,
|
typedef int (*COLLATE)(ntfs_volume *vol, const void *data1, int len1, const void *data2, int len2);
|
||||||
const void *data2, int len2);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct ntfs_index_context -
|
* struct ntfs_index_context -
|
||||||
@ -112,42 +111,38 @@ typedef int (*COLLATE)(ntfs_volume *vol, const void *data1, int len1,
|
|||||||
* the call to ntfs_index_ctx_put() to ensure that the changes are written
|
* the call to ntfs_index_ctx_put() to ensure that the changes are written
|
||||||
* to disk.
|
* to disk.
|
||||||
*/
|
*/
|
||||||
typedef struct {
|
typedef struct
|
||||||
ntfs_inode *ni;
|
{
|
||||||
ntfschar *name;
|
ntfs_inode *ni;
|
||||||
u32 name_len;
|
ntfschar *name;
|
||||||
INDEX_ENTRY *entry;
|
u32 name_len;
|
||||||
void *data;
|
INDEX_ENTRY *entry;
|
||||||
u16 data_len;
|
void *data;
|
||||||
COLLATE collate;
|
u16 data_len;
|
||||||
BOOL is_in_root;
|
COLLATE collate;
|
||||||
INDEX_ROOT *ir;
|
BOOL is_in_root;
|
||||||
ntfs_attr_search_ctx *actx;
|
INDEX_ROOT *ir;
|
||||||
INDEX_BLOCK *ib;
|
ntfs_attr_search_ctx *actx;
|
||||||
ntfs_attr *ia_na;
|
INDEX_BLOCK *ib;
|
||||||
int parent_pos[MAX_PARENT_VCN]; /* parent entries' positions */
|
ntfs_attr *ia_na;
|
||||||
VCN parent_vcn[MAX_PARENT_VCN]; /* entry's parent nodes */
|
int parent_pos[MAX_PARENT_VCN]; /* parent entries' positions */
|
||||||
int pindex; /* maximum it's the number of the parent nodes */
|
VCN parent_vcn[MAX_PARENT_VCN]; /* entry's parent nodes */
|
||||||
BOOL ib_dirty;
|
int pindex; /* maximum it's the number of the parent nodes */
|
||||||
u32 block_size;
|
BOOL ib_dirty;
|
||||||
u8 vcn_size_bits;
|
u32 block_size;
|
||||||
|
u8 vcn_size_bits;
|
||||||
} ntfs_index_context;
|
} ntfs_index_context;
|
||||||
|
|
||||||
extern ntfs_index_context *ntfs_index_ctx_get(ntfs_inode *ni,
|
extern ntfs_index_context *ntfs_index_ctx_get(ntfs_inode *ni, ntfschar *name, u32 name_len);
|
||||||
ntfschar *name, u32 name_len);
|
|
||||||
extern void ntfs_index_ctx_put(ntfs_index_context *ictx);
|
extern void ntfs_index_ctx_put(ntfs_index_context *ictx);
|
||||||
extern void ntfs_index_ctx_reinit(ntfs_index_context *ictx);
|
extern void ntfs_index_ctx_reinit(ntfs_index_context *ictx);
|
||||||
|
|
||||||
extern int ntfs_index_lookup(const void *key, const int key_len,
|
extern int ntfs_index_lookup(const void *key, const int key_len, ntfs_index_context *ictx) __attribute_warn_unused_result__;
|
||||||
ntfs_index_context *ictx) __attribute_warn_unused_result__;
|
|
||||||
|
|
||||||
extern INDEX_ENTRY *ntfs_index_next(INDEX_ENTRY *ie,
|
extern INDEX_ENTRY *ntfs_index_next(INDEX_ENTRY *ie, ntfs_index_context *ictx);
|
||||||
ntfs_index_context *ictx);
|
|
||||||
|
|
||||||
extern int ntfs_index_add_filename(ntfs_inode *ni, FILE_NAME_ATTR *fn,
|
extern int ntfs_index_add_filename(ntfs_inode *ni, FILE_NAME_ATTR *fn, MFT_REF mref);
|
||||||
MFT_REF mref);
|
extern int ntfs_index_remove(ntfs_inode *dir_ni, ntfs_inode *ni, const void *key, const int keylen);
|
||||||
extern int ntfs_index_remove(ntfs_inode *dir_ni, ntfs_inode *ni,
|
|
||||||
const void *key, const int keylen);
|
|
||||||
|
|
||||||
extern INDEX_ROOT *ntfs_index_root_get(ntfs_inode *ni, ATTR_RECORD *attr);
|
extern INDEX_ROOT *ntfs_index_root_get(ntfs_inode *ni, ATTR_RECORD *attr);
|
||||||
|
|
||||||
|
File diff suppressed because it is too large
Load Diff
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user