Compare commits
20 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| b7775d3d82 | |||
| bf110e5341 | |||
| 9183815634 | |||
| c6e33e5285 | |||
| 709cfbf750 | |||
| 5c02645036 | |||
| 75e4aaa3b4 | |||
| f2384e93c7 | |||
| 7fdf5751b1 | |||
| 0402fac389 | |||
| 17f09ab84a | |||
| 9328113fcd | |||
| 4fde40ece9 | |||
| 5e80cc80b8 | |||
| 9f3b338fed | |||
| 0052cccc14 | |||
| e25d4a67da | |||
| 78edb14d76 | |||
| 416883b354 | |||
| 8eb73298f5 |
+3
-1
@@ -7,9 +7,11 @@ include(address_sanitizer)
|
|||||||
|
|
||||||
set(OX_BUILD_EXEC "ON" CACHE STRING "Build executables (ON/OFF)")
|
set(OX_BUILD_EXEC "ON" CACHE STRING "Build executables (ON/OFF)")
|
||||||
set(OX_RUN_TESTS "ON" CACHE STRING "Run tests (ON/OFF)")
|
set(OX_RUN_TESTS "ON" CACHE STRING "Run tests (ON/OFF)")
|
||||||
|
set(OX_USE_STDLIB "ON" CACHE STRING "Build libraries that need the std lib (ON/OFF)")
|
||||||
|
|
||||||
# can't run tests without building them
|
# can't run tests without building them
|
||||||
if(OX_BUILD_EXEC STREQUAL "OFF")
|
if(OX_BUILD_EXEC STREQUAL "OFF" OR OX_USE_STDLIB STREQUAL "OFF")
|
||||||
|
set(OX_BUILD_EXEC "OFF")
|
||||||
set(OX_RUN_TESTS "OFF")
|
set(OX_RUN_TESTS "OFF")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
|||||||
+9
-7
@@ -1,9 +1,11 @@
|
|||||||
if(${CMAKE_FIND_ROOT_PATH})
|
if("${CMAKE_FIND_ROOT_PATH}" STREQUAL "")
|
||||||
set(Ox_INCLUDE_DIRS ${CMAKE_FIND_ROOT_PATH}/include/)
|
|
||||||
set(OxStd_LIBRARY ${CMAKE_FIND_ROOT_PATH}/lib/ox/libOxStd.a)
|
|
||||||
set(OxFs_LIBRARY ${CMAKE_FIND_ROOT_PATH}/lib/ox/libOxFs.a)
|
|
||||||
else(${CMAKE_FIND_ROOT_PATH})
|
|
||||||
set(Ox_INCLUDE_DIRS /usr/local/include/)
|
set(Ox_INCLUDE_DIRS /usr/local/include/)
|
||||||
set(OxStd_LIBRARY /usr/local/lib/ox/libOxStd.a)
|
set(OxStd_LIBRARY /usr/local/lib/ox/libOxStd.a)
|
||||||
set(OxFs_LIBRARY /usr/local/lib/ox/libOxFs.a)
|
set(OxFS_LIBRARY /usr/local/lib/ox/libOxFS.a)
|
||||||
endif(${CMAKE_FIND_ROOT_PATH})
|
set(OxClArgs_LIBRARY /usr/local/lib/ox/libOxClArgs.a)
|
||||||
|
else("${CMAKE_FIND_ROOT_PATH}" STREQUAL "")
|
||||||
|
set(Ox_INCLUDE_DIRS ${CMAKE_FIND_ROOT_PATH}/include/)
|
||||||
|
set(OxStd_LIBRARY ${CMAKE_FIND_ROOT_PATH}/lib/ox/libOxStd.a)
|
||||||
|
set(OxFS_LIBRARY ${CMAKE_FIND_ROOT_PATH}/lib/ox/libOxFS.a)
|
||||||
|
set(OxClArgs_LIBRARY ${CMAKE_FIND_ROOT_PATH}/lib/ox/libOxClArgs.a)
|
||||||
|
endif("${CMAKE_FIND_ROOT_PATH}" STREQUAL "")
|
||||||
|
|||||||
+1
-1
@@ -10,7 +10,7 @@ BUILD_TYPE=$2
|
|||||||
if [[ $TARGET == windows ]]; then
|
if [[ $TARGET == windows ]]; then
|
||||||
toolchain="-DCMAKE_TOOLCHAIN_FILE=cmake/Modules/Mingw.cmake"
|
toolchain="-DCMAKE_TOOLCHAIN_FILE=cmake/Modules/Mingw.cmake"
|
||||||
elif [[ $TARGET == gba ]]; then
|
elif [[ $TARGET == gba ]]; then
|
||||||
toolchain="-DCMAKE_TOOLCHAIN_FILE=cmake/Modules/GBA.cmake -DOX_BUILD_EXEC=OFF"
|
toolchain="-DCMAKE_TOOLCHAIN_FILE=cmake/Modules/GBA.cmake -DOX_USE_STDLIB=OFF"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ $BUILD_TYPE == debug ]]; then
|
if [[ $BUILD_TYPE == debug ]]; then
|
||||||
|
|||||||
@@ -1,4 +1,7 @@
|
|||||||
cmake_minimum_required(VERSION 2.8)
|
cmake_minimum_required(VERSION 2.8)
|
||||||
|
|
||||||
|
if(OX_USE_STDLIB STREQUAL "ON")
|
||||||
|
add_subdirectory(clargs)
|
||||||
|
endif(OX_USE_STDLIB STREQUAL "ON")
|
||||||
add_subdirectory(fs)
|
add_subdirectory(fs)
|
||||||
add_subdirectory(std)
|
add_subdirectory(std)
|
||||||
|
|||||||
@@ -0,0 +1,20 @@
|
|||||||
|
cmake_minimum_required(VERSION 2.8)
|
||||||
|
|
||||||
|
add_library(
|
||||||
|
OxClArgs
|
||||||
|
clargs.cpp
|
||||||
|
)
|
||||||
|
|
||||||
|
install(
|
||||||
|
FILES
|
||||||
|
clargs.hpp
|
||||||
|
DESTINATION
|
||||||
|
include/ox/clargs
|
||||||
|
)
|
||||||
|
|
||||||
|
install(
|
||||||
|
TARGETS
|
||||||
|
OxClArgs
|
||||||
|
LIBRARY DESTINATION lib/ox
|
||||||
|
ARCHIVE DESTINATION lib/ox
|
||||||
|
)
|
||||||
@@ -0,0 +1,55 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2015 - 2017 gtalent2@gmail.com
|
||||||
|
*
|
||||||
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
||||||
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <ox/std/strops.hpp>
|
||||||
|
#include "clargs.hpp"
|
||||||
|
|
||||||
|
namespace ox {
|
||||||
|
namespace clargs {
|
||||||
|
|
||||||
|
using namespace ::std;
|
||||||
|
|
||||||
|
ClArgs::ClArgs(int argc, const char **args) {
|
||||||
|
for (int i = 0; i < argc; i++) {
|
||||||
|
string arg = args[i];
|
||||||
|
if (arg[0] == '-') {
|
||||||
|
while (arg[0] == '-' && arg.size()) {
|
||||||
|
arg = arg.substr(1);
|
||||||
|
}
|
||||||
|
m_bools[arg] = true;
|
||||||
|
|
||||||
|
// parse additional arguments
|
||||||
|
if (i < argc && args[i + 1]) {
|
||||||
|
string val = args[i + 1];
|
||||||
|
if (val.size() && val[i] != '-') {
|
||||||
|
if (val == "false") {
|
||||||
|
m_bools[arg] = false;
|
||||||
|
}
|
||||||
|
m_strings[arg] = val;
|
||||||
|
m_ints[arg] = ox_atoi(val.c_str());
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ClArgs::getBool(const char *arg) {
|
||||||
|
return m_bools[arg];
|
||||||
|
}
|
||||||
|
|
||||||
|
string ClArgs::getString(const char *arg) {
|
||||||
|
return m_strings[arg];
|
||||||
|
}
|
||||||
|
|
||||||
|
int ClArgs::getInt(const char *arg) {
|
||||||
|
return m_ints[arg];
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,34 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2015 - 2017 gtalent2@gmail.com
|
||||||
|
*
|
||||||
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
||||||
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <map>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
namespace ox {
|
||||||
|
namespace clargs {
|
||||||
|
|
||||||
|
class ClArgs {
|
||||||
|
private:
|
||||||
|
::std::map<::std::string, bool> m_bools;
|
||||||
|
::std::map<::std::string, ::std::string> m_strings;
|
||||||
|
::std::map<::std::string, int> m_ints;
|
||||||
|
|
||||||
|
public:
|
||||||
|
ClArgs(int argc, const char **args);
|
||||||
|
|
||||||
|
bool getBool(const char *arg);
|
||||||
|
|
||||||
|
::std::string getString(const char *arg);
|
||||||
|
|
||||||
|
int getInt(const char *arg);
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
+73
-49
@@ -103,7 +103,7 @@ class FileStore {
|
|||||||
|
|
||||||
struct StatInfo {
|
struct StatInfo {
|
||||||
InodeId_t inodeId;
|
InodeId_t inodeId;
|
||||||
typename Header::FsSize_t size;
|
typename Header::FsSize_t size;
|
||||||
uint8_t fileType;
|
uint8_t fileType;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -143,7 +143,7 @@ class FileStore {
|
|||||||
typename Header::FsSize_t getRight();
|
typename Header::FsSize_t getRight();
|
||||||
|
|
||||||
void setData(void *data, typename Header::FsSize_t size);
|
void setData(void *data, typename Header::FsSize_t size);
|
||||||
void *getData();
|
uint8_t *getData();
|
||||||
};
|
};
|
||||||
|
|
||||||
Header m_header;
|
Header m_header;
|
||||||
@@ -186,6 +186,20 @@ class FileStore {
|
|||||||
*/
|
*/
|
||||||
int read(InodeId_t id, void *data, typename Header::FsSize_t *size);
|
int read(InodeId_t id, void *data, typename Header::FsSize_t *size);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the "file" at the given id. You are responsible for freeing
|
||||||
|
* the data when done with it.
|
||||||
|
* @param id id of the "file"
|
||||||
|
* @param readStart where in the data to start reading
|
||||||
|
* @param readSize how much data to read
|
||||||
|
* @param data pointer to the pointer where the data is stored
|
||||||
|
* @param size pointer to a value that will be assigned the size of data
|
||||||
|
* @return 0 if read is a success
|
||||||
|
*/
|
||||||
|
int read(InodeId_t id, typename Header::FsSize_t readStart,
|
||||||
|
typename Header::FsSize_t readSize, void *data,
|
||||||
|
typename Header::FsSize_t *size);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Reads the stat information of the inode of the given inode id.
|
* Reads the stat information of the inode of the given inode id.
|
||||||
* If the returned inode id is 0, then the requested inode was not found.
|
* If the returned inode id is 0, then the requested inode was not found.
|
||||||
@@ -200,7 +214,7 @@ class FileStore {
|
|||||||
* @param size the size of the data to insert
|
* @param size the size of the data to insert
|
||||||
* @return the space currently available in this file store.
|
* @return the space currently available in this file store.
|
||||||
*/
|
*/
|
||||||
typename Header::FsSize_t spaceNeeded(InodeId_t id, typename Header::FsSize_t size);
|
typename Header::FsSize_t spaceNeeded(typename Header::FsSize_t size);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the size of the file store.
|
* Returns the size of the file store.
|
||||||
@@ -240,6 +254,20 @@ class FileStore {
|
|||||||
*/
|
*/
|
||||||
Inode *getInodeParent(Inode *root, InodeId_t id, typename Header::FsSize_t targetAddr);
|
Inode *getInodeParent(Inode *root, InodeId_t id, typename Header::FsSize_t targetAddr);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the "file" at the given id. You are responsible for freeing
|
||||||
|
* the data when done with it.
|
||||||
|
* @param inode inode of the "file"
|
||||||
|
* @param readStart where in the data to start reading
|
||||||
|
* @param readSize how much data to read
|
||||||
|
* @param data pointer to the pointer where the data is stored
|
||||||
|
* @param size pointer to a value that will be assigned the size of data
|
||||||
|
* @return 0 if read is a success
|
||||||
|
*/
|
||||||
|
int read(Inode *inode, typename Header::FsSize_t readStart,
|
||||||
|
typename Header::FsSize_t readSize, void *data,
|
||||||
|
typename Header::FsSize_t *size);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Removes the inode of the given ID.
|
* Removes the inode of the given ID.
|
||||||
* @param id the id of the file
|
* @param id the id of the file
|
||||||
@@ -276,15 +304,9 @@ class FileStore {
|
|||||||
*/
|
*/
|
||||||
bool insert(Inode *root, Inode *insertValue);
|
bool insert(Inode *root, Inode *insertValue);
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the FsSize_t associated with the next Inode to be allocated.
|
|
||||||
* @retrun the FsSize_t associated with the next Inode to be allocated
|
|
||||||
*/
|
|
||||||
typename Header::FsSize_t iterator();
|
|
||||||
|
|
||||||
typename Header::FsSize_t firstInode();
|
typename Header::FsSize_t firstInode();
|
||||||
|
|
||||||
Inode *lastInode();
|
typename Header::FsSize_t lastInode();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Updates the address of the inode in the tree.
|
* Updates the address of the inode in the tree.
|
||||||
@@ -326,7 +348,7 @@ void FileStore<Header>::Inode::setDataLen(typename Header::FsSize_t dataLen) {
|
|||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
typename Header::FsSize_t FileStore<Header>::Inode::getDataLen() {
|
typename Header::FsSize_t FileStore<Header>::Inode::getDataLen() {
|
||||||
return std::bigEndianAdapt(this->m_dataLen);
|
return std::bigEndianAdapt(m_dataLen);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
@@ -336,7 +358,7 @@ void FileStore<Header>::Inode::setPrev(typename Header::FsSize_t prev) {
|
|||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
typename Header::FsSize_t FileStore<Header>::Inode::getPrev() {
|
typename Header::FsSize_t FileStore<Header>::Inode::getPrev() {
|
||||||
return std::bigEndianAdapt(this->m_prev);
|
return std::bigEndianAdapt(m_prev);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
@@ -346,7 +368,7 @@ void FileStore<Header>::Inode::setNext(typename Header::FsSize_t next) {
|
|||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
typename Header::FsSize_t FileStore<Header>::Inode::getNext() {
|
typename Header::FsSize_t FileStore<Header>::Inode::getNext() {
|
||||||
return std::bigEndianAdapt(this->m_next);
|
return std::bigEndianAdapt(m_next);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
@@ -356,7 +378,7 @@ void FileStore<Header>::Inode::setId(InodeId_t id) {
|
|||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
typename Header::InodeId_t FileStore<Header>::Inode::getId() {
|
typename Header::InodeId_t FileStore<Header>::Inode::getId() {
|
||||||
return std::bigEndianAdapt(this->m_id);
|
return std::bigEndianAdapt(m_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
@@ -366,7 +388,7 @@ void FileStore<Header>::Inode::setFileType(uint8_t fileType) {
|
|||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
uint8_t FileStore<Header>::Inode::getFileType() {
|
uint8_t FileStore<Header>::Inode::getFileType() {
|
||||||
return std::bigEndianAdapt(this->m_fileType);
|
return std::bigEndianAdapt(m_fileType);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
@@ -376,7 +398,7 @@ void FileStore<Header>::Inode::setLeft(typename Header::FsSize_t left) {
|
|||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
typename Header::FsSize_t FileStore<Header>::Inode::getLeft() {
|
typename Header::FsSize_t FileStore<Header>::Inode::getLeft() {
|
||||||
return std::bigEndianAdapt(this->m_left);
|
return std::bigEndianAdapt(m_left);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
@@ -386,19 +408,19 @@ void FileStore<Header>::Inode::setRight(typename Header::FsSize_t right) {
|
|||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
typename Header::FsSize_t FileStore<Header>::Inode::getRight() {
|
typename Header::FsSize_t FileStore<Header>::Inode::getRight() {
|
||||||
return std::bigEndianAdapt(this->m_right);
|
return std::bigEndianAdapt(m_right);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
void FileStore<Header>::Inode::setData(void *data, typename Header::FsSize_t size) {
|
void FileStore<Header>::Inode::setData(void *data, typename Header::FsSize_t size) {
|
||||||
ox_memcpy(this->getData(), data, size);
|
ox_memcpy(getData(), data, size);
|
||||||
setDataLen(size);
|
setDataLen(size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
void *FileStore<Header>::Inode::getData() {
|
uint8_t *FileStore<Header>::Inode::getData() {
|
||||||
return this + 1;
|
return (uint8_t*) (this + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -544,15 +566,28 @@ void FileStore<Header>::updateInodeAddress(InodeId_t id, typename Header::FsSize
|
|||||||
template<typename Header>
|
template<typename Header>
|
||||||
int FileStore<Header>::read(InodeId_t id, void *data, typename Header::FsSize_t *size) {
|
int FileStore<Header>::read(InodeId_t id, void *data, typename Header::FsSize_t *size) {
|
||||||
auto inode = getInode(ptr<Inode*>(m_header.getRootInode()), id);
|
auto inode = getInode(ptr<Inode*>(m_header.getRootInode()), id);
|
||||||
int retval = 1;
|
return inode ? read(inode, 0, inode->getDataLen(), data, size) : 1;
|
||||||
if (inode) {
|
}
|
||||||
if (size) {
|
|
||||||
*size = inode->getDataLen();
|
template<typename Header>
|
||||||
}
|
int FileStore<Header>::read(InodeId_t id, typename Header::FsSize_t readStart,
|
||||||
ox_memcpy(data, inode->getData(), inode->getDataLen());
|
typename Header::FsSize_t readSize, void *data, typename Header::FsSize_t *size) {
|
||||||
retval = 0;
|
auto inode = getInode(ptr<Inode*>(m_header.getRootInode()), id);
|
||||||
|
return inode ? read(inode, readStart, readSize, data, size) : 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename Header>
|
||||||
|
int FileStore<Header>::read(Inode *inode, typename Header::FsSize_t readStart,
|
||||||
|
typename Header::FsSize_t readSize, void *data, typename Header::FsSize_t *size) {
|
||||||
|
// be sure read size is not greater than what is available to read
|
||||||
|
if (inode->getDataLen() - readStart < readSize) {
|
||||||
|
readSize = inode->getDataLen() - readStart;
|
||||||
}
|
}
|
||||||
return retval;
|
if (size) {
|
||||||
|
*size = readSize;
|
||||||
|
}
|
||||||
|
ox_memcpy(data, inode->getData() + readStart, readSize);
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
@@ -570,13 +605,8 @@ typename FileStore<Header>::StatInfo FileStore<Header>::stat(InodeId_t id) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
typename Header::FsSize_t FileStore<Header>::spaceNeeded(InodeId_t id, typename Header::FsSize_t size) {
|
typename Header::FsSize_t FileStore<Header>::spaceNeeded(typename Header::FsSize_t size) {
|
||||||
typename Header::FsSize_t needed = sizeof(Inode) + size;;
|
return sizeof(Inode) + size;
|
||||||
auto inode = getInode(ptr<Inode*>(m_header.getRootInode()), id);
|
|
||||||
if (inode) {
|
|
||||||
needed -= inode->size();
|
|
||||||
}
|
|
||||||
return needed;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
@@ -635,17 +665,16 @@ typename FileStore<Header>::Inode *FileStore<Header>::getInodeParent(Inode *root
|
|||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
typename Header::FsSize_t FileStore<Header>::nextInodeAddr() {
|
typename Header::FsSize_t FileStore<Header>::nextInodeAddr() {
|
||||||
typename Header::FsSize_t next = ptr(lastInode()) + lastInode()->size();
|
return lastInode() + ptr<Inode*>(lastInode())->size();
|
||||||
return next;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
void *FileStore<Header>::alloc(typename Header::FsSize_t size) {
|
void *FileStore<Header>::alloc(typename Header::FsSize_t size) {
|
||||||
typename Header::FsSize_t next = nextInodeAddr();
|
auto next = nextInodeAddr();
|
||||||
if ((next + size) > (uint64_t) end()) {
|
if ((next + size) > ptr(end())) {
|
||||||
compact();
|
compact();
|
||||||
next = nextInodeAddr();
|
next = nextInodeAddr();
|
||||||
if ((next + size) > (uint64_t) end()) {
|
if ((next + size) > ptr(end())) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -654,7 +683,7 @@ void *FileStore<Header>::alloc(typename Header::FsSize_t size) {
|
|||||||
const auto inode = ptr<Inode*>(retval);
|
const auto inode = ptr<Inode*>(retval);
|
||||||
ox_memset(inode, 0, size);
|
ox_memset(inode, 0, size);
|
||||||
inode->setPrev(ptr<Inode*>(firstInode())->getPrev());
|
inode->setPrev(ptr<Inode*>(firstInode())->getPrev());
|
||||||
inode->setNext(retval + size);
|
inode->setNext(firstInode());
|
||||||
m_header.setMemUsed(m_header.getMemUsed() + size);
|
m_header.setMemUsed(m_header.getMemUsed() + size);
|
||||||
ptr<Inode*>(firstInode())->setPrev(retval);
|
ptr<Inode*>(firstInode())->setPrev(retval);
|
||||||
return inode;
|
return inode;
|
||||||
@@ -702,11 +731,6 @@ bool FileStore<Header>::insert(Inode *root, Inode *insertValue) {
|
|||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
|
||||||
typename Header::FsSize_t FileStore<Header>::iterator() {
|
|
||||||
return ptr(lastInode()) + lastInode()->size();
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
typename Header::FsSize_t FileStore<Header>::ptr(void *ptr) {
|
typename Header::FsSize_t FileStore<Header>::ptr(void *ptr) {
|
||||||
#ifdef _MSC_VER
|
#ifdef _MSC_VER
|
||||||
@@ -724,8 +748,8 @@ typename Header::FsSize_t FileStore<Header>::firstInode() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
typename FileStore<Header>::Inode *FileStore<Header>::lastInode() {
|
typename Header::FsSize_t FileStore<Header>::lastInode() {
|
||||||
return ptr<Inode*>(ptr<Inode*>(firstInode())->getPrev());
|
return ptr<Inode*>(firstInode())->getPrev();
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Header>
|
template<typename Header>
|
||||||
@@ -749,7 +773,7 @@ uint8_t *FileStore<Header>::format(uint8_t *buffer, typename Header::FsSize_t si
|
|||||||
fs->m_header.setMemUsed(sizeof(FileStore<Header>) + sizeof(Inode));
|
fs->m_header.setMemUsed(sizeof(FileStore<Header>) + sizeof(Inode));
|
||||||
fs->m_header.setRootInode(sizeof(FileStore<Header>));
|
fs->m_header.setRootInode(sizeof(FileStore<Header>));
|
||||||
((Inode*) (fs + 1))->setPrev(sizeof(FileStore<Header>));
|
((Inode*) (fs + 1))->setPrev(sizeof(FileStore<Header>));
|
||||||
fs->lastInode()->setNext(sizeof(FileStore<Header>));
|
((Inode*) (fs + 1))->setNext(sizeof(FileStore<Header>));
|
||||||
|
|
||||||
return (uint8_t*) buffer;
|
return (uint8_t*) buffer;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,7 +10,7 @@
|
|||||||
namespace ox {
|
namespace ox {
|
||||||
namespace fs {
|
namespace fs {
|
||||||
|
|
||||||
FileSystem *createFileSystem(void *buff) {
|
FileSystem *createFileSystem(void *buff, size_t buffSize) {
|
||||||
auto version = ((FileStore16*) buff)->version();
|
auto version = ((FileStore16*) buff)->version();
|
||||||
auto type = ((FileStore16*) buff)->fsType();
|
auto type = ((FileStore16*) buff)->fsType();
|
||||||
FileSystem *fs = nullptr;
|
FileSystem *fs = nullptr;
|
||||||
@@ -33,8 +33,42 @@ FileSystem *createFileSystem(void *buff) {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (fs && fs->size() > buffSize) {
|
||||||
|
delete fs;
|
||||||
|
fs = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
return fs;
|
return fs;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
FileSystem *expandCopy(FileSystem *fs, size_t size) {
|
||||||
|
auto fsBuff = fs->buff();
|
||||||
|
FileSystem *retval = nullptr;
|
||||||
|
|
||||||
|
if (fs->size() <= size) {
|
||||||
|
auto cloneBuff = new uint8_t[size];
|
||||||
|
ox_memcpy(cloneBuff, fsBuff, fs->size());
|
||||||
|
|
||||||
|
fsBuff = cloneBuff;
|
||||||
|
retval = createFileSystem(fsBuff, size);
|
||||||
|
retval->resize(size);
|
||||||
|
}
|
||||||
|
|
||||||
|
return retval;
|
||||||
|
}
|
||||||
|
|
||||||
|
FileSystem *expandCopyCleanup(FileSystem *fs, size_t size) {
|
||||||
|
auto out = expandCopy(fs, size);
|
||||||
|
|
||||||
|
if (out) {
|
||||||
|
delete[] fs->buff();
|
||||||
|
delete fs;
|
||||||
|
} else {
|
||||||
|
out = fs;
|
||||||
|
}
|
||||||
|
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+54
-13
@@ -36,6 +36,8 @@ class FileSystem {
|
|||||||
|
|
||||||
virtual int read(uint64_t inode, void *buffer, size_t size) = 0;
|
virtual int read(uint64_t inode, void *buffer, size_t size) = 0;
|
||||||
|
|
||||||
|
virtual int read(uint64_t inode, size_t readStart, size_t readSize, void *buffer, size_t *size) = 0;
|
||||||
|
|
||||||
virtual uint8_t *read(uint64_t inode, size_t *size) = 0;
|
virtual uint8_t *read(uint64_t inode, size_t *size) = 0;
|
||||||
|
|
||||||
virtual int remove(uint64_t inode) = 0;
|
virtual int remove(uint64_t inode) = 0;
|
||||||
@@ -46,14 +48,27 @@ class FileSystem {
|
|||||||
|
|
||||||
virtual FileStat stat(uint64_t inode) = 0;
|
virtual FileStat stat(uint64_t inode) = 0;
|
||||||
|
|
||||||
virtual uint64_t spaceNeeded(uint64_t id, uint64_t size) = 0;
|
virtual uint64_t spaceNeeded(uint64_t size) = 0;
|
||||||
|
|
||||||
virtual uint64_t available() = 0;
|
virtual uint64_t available() = 0;
|
||||||
|
|
||||||
virtual uint64_t size() = 0;
|
virtual uint64_t size() = 0;
|
||||||
|
|
||||||
|
virtual uint8_t *buff() = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
FileSystem *createFileSystem(void *buff);
|
FileSystem *createFileSystem(void *buff, size_t buffSize);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a larger version of the given FileSystem.
|
||||||
|
*/
|
||||||
|
FileSystem *expandCopy(FileSystem *src);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Calls expandCopy and deletes the original FileSystem and buff a resize was
|
||||||
|
* performed.
|
||||||
|
*/
|
||||||
|
FileSystem *expandCopyCleanup(FileSystem *fs, size_t size);
|
||||||
|
|
||||||
template<typename FileStore, FsType FS_TYPE>
|
template<typename FileStore, FsType FS_TYPE>
|
||||||
class FileSystemTemplate: public FileSystem {
|
class FileSystemTemplate: public FileSystem {
|
||||||
@@ -97,9 +112,11 @@ class FileSystemTemplate: public FileSystem {
|
|||||||
|
|
||||||
int read(const char *path, void *buffer);
|
int read(const char *path, void *buffer);
|
||||||
|
|
||||||
uint8_t *read(uint64_t inode, size_t *size) override;
|
int read(uint64_t inode, void *buffer, size_t buffSize) override;
|
||||||
|
|
||||||
int read(uint64_t inode, void *buffer, size_t size) override;
|
int read(uint64_t inode, size_t readStart, size_t readSize, void *buffer, size_t *size) override;
|
||||||
|
|
||||||
|
uint8_t *read(uint64_t inode, size_t *size) override;
|
||||||
|
|
||||||
void resize(uint64_t size = 0) override;
|
void resize(uint64_t size = 0) override;
|
||||||
|
|
||||||
@@ -111,12 +128,14 @@ class FileSystemTemplate: public FileSystem {
|
|||||||
|
|
||||||
FileStat stat(uint64_t inode) override;
|
FileStat stat(uint64_t inode) override;
|
||||||
|
|
||||||
uint64_t spaceNeeded(uint64_t id, uint64_t size) override;
|
uint64_t spaceNeeded(uint64_t size) override;
|
||||||
|
|
||||||
uint64_t available() override;
|
uint64_t available() override;
|
||||||
|
|
||||||
uint64_t size() override;
|
uint64_t size() override;
|
||||||
|
|
||||||
|
uint8_t *buff() override;
|
||||||
|
|
||||||
static uint8_t *format(void *buffer, typename FileStore::FsSize_t size, bool useDirectories);
|
static uint8_t *format(void *buffer, typename FileStore::FsSize_t size, bool useDirectories);
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -159,18 +178,35 @@ FileStat FileSystemTemplate<FileStore, FS_TYPE>::stat(uint64_t inode) {
|
|||||||
#pragma warning(disable:4244)
|
#pragma warning(disable:4244)
|
||||||
#endif
|
#endif
|
||||||
template<typename FileStore, FsType FS_TYPE>
|
template<typename FileStore, FsType FS_TYPE>
|
||||||
int FileSystemTemplate<FileStore, FS_TYPE>::read(uint64_t inode, void *buffer, size_t size) {
|
int FileSystemTemplate<FileStore, FS_TYPE>::read(uint64_t inode, void *buffer, size_t buffSize) {
|
||||||
auto err = 1;
|
auto stat = store->stat(inode);
|
||||||
auto s = store->stat(inode);
|
if (stat.size <= buffSize) {
|
||||||
if (size == s.size) {
|
return store->read(inode, buffer, nullptr);
|
||||||
err = store->read(inode, buffer, nullptr);
|
|
||||||
}
|
}
|
||||||
return err;
|
return 0;
|
||||||
|
;
|
||||||
}
|
}
|
||||||
#ifdef _MSC_VER
|
#ifdef _MSC_VER
|
||||||
#pragma warning(default:4244)
|
#pragma warning(default:4244)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef _MSC_VER
|
||||||
|
#pragma warning(disable:4244)
|
||||||
|
#endif
|
||||||
|
template<typename FileStore, FsType FS_TYPE>
|
||||||
|
int FileSystemTemplate<FileStore, FS_TYPE>::read(uint64_t inode, size_t readStart,
|
||||||
|
size_t readSize, void *buffer,
|
||||||
|
size_t *size) {
|
||||||
|
if (size) {
|
||||||
|
auto stat = store->stat(inode);
|
||||||
|
*size = stat.size;
|
||||||
|
}
|
||||||
|
return store->read(inode, readStart, readSize, buffer, nullptr);
|
||||||
|
}
|
||||||
|
#ifdef _MSC_VER
|
||||||
|
#pragma warning(disable:4244)
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef _MSC_VER
|
#ifdef _MSC_VER
|
||||||
#pragma warning(disable:4244)
|
#pragma warning(disable:4244)
|
||||||
#endif
|
#endif
|
||||||
@@ -219,8 +255,8 @@ void FileSystemTemplate<FileStore, FS_TYPE>::resize(uint64_t size) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
template<typename FileStore, FsType FS_TYPE>
|
template<typename FileStore, FsType FS_TYPE>
|
||||||
uint64_t FileSystemTemplate<FileStore, FS_TYPE>::spaceNeeded(uint64_t id, uint64_t size) {
|
uint64_t FileSystemTemplate<FileStore, FS_TYPE>::spaceNeeded(uint64_t size) {
|
||||||
return store->spaceNeeded(id, size);
|
return store->spaceNeeded(size);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename FileStore, FsType FS_TYPE>
|
template<typename FileStore, FsType FS_TYPE>
|
||||||
@@ -233,6 +269,11 @@ uint64_t FileSystemTemplate<FileStore, FS_TYPE>::size() {
|
|||||||
return store->size();
|
return store->size();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename FileStore, FsType FS_TYPE>
|
||||||
|
uint8_t *FileSystemTemplate<FileStore, FS_TYPE>::buff() {
|
||||||
|
return (uint8_t*) store;
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef _MSC_VER
|
#ifdef _MSC_VER
|
||||||
#pragma warning(disable:4244)
|
#pragma warning(disable:4244)
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
+30
-31
@@ -30,13 +30,12 @@ const static auto usage = "usage:\n"
|
|||||||
"\toxfs compact <FS file>\n"
|
"\toxfs compact <FS file>\n"
|
||||||
"\toxfs version\n";
|
"\toxfs version\n";
|
||||||
|
|
||||||
char *loadFileBuff(const char *path, ::size_t *sizeOut = nullptr) {
|
char *loadFileBuff(FILE *file, ::size_t *sizeOut = nullptr) {
|
||||||
auto file = fopen(path, "rb");
|
|
||||||
if (file) {
|
if (file) {
|
||||||
fseek(file, 0, SEEK_END);
|
fseek(file, 0, SEEK_END);
|
||||||
const auto size = ftell(file);
|
const auto size = ftell(file);
|
||||||
rewind(file);
|
rewind(file);
|
||||||
auto buff = (char*) malloc(size);
|
auto buff = new char[size];
|
||||||
auto itemsRead = fread(buff, size, 1, file);
|
auto itemsRead = fread(buff, size, 1, file);
|
||||||
fclose(file);
|
fclose(file);
|
||||||
if (sizeOut) {
|
if (sizeOut) {
|
||||||
@@ -48,12 +47,17 @@ char *loadFileBuff(const char *path, ::size_t *sizeOut = nullptr) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
char *loadFileBuff(const char *path, ::size_t *sizeOut = nullptr) {
|
||||||
|
return loadFileBuff(fopen(path, "rb"), sizeOut);
|
||||||
|
}
|
||||||
|
|
||||||
size_t bytes(const char *str) {
|
size_t bytes(const char *str) {
|
||||||
auto size = ::ox_strlen(str);
|
auto size = ::ox_strlen(str);
|
||||||
const auto lastChar = str[size-1];
|
const auto lastChar = str[size-1];
|
||||||
auto multiplier = 1;
|
auto multiplier = 1;
|
||||||
auto copy = new char[size];
|
char copy[size + 1];
|
||||||
ox_memcpy(copy, str, size);
|
ox_memcpy(copy, str, size + 1);
|
||||||
|
// parse size unit
|
||||||
if (lastChar < '0' || lastChar > '9') {
|
if (lastChar < '0' || lastChar > '9') {
|
||||||
copy[size-1] = 0;
|
copy[size-1] = 0;
|
||||||
switch (lastChar) {
|
switch (lastChar) {
|
||||||
@@ -73,9 +77,7 @@ size_t bytes(const char *str) {
|
|||||||
multiplier = -1;
|
multiplier = -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const auto retval = ((size_t) ox_atoi(copy)) * multiplier;
|
return ox_atoi(copy) * multiplier;
|
||||||
delete []copy;
|
|
||||||
return retval;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int format(int argc, char **args) {
|
int format(int argc, char **args) {
|
||||||
@@ -83,13 +85,10 @@ int format(int argc, char **args) {
|
|||||||
auto err = 0;
|
auto err = 0;
|
||||||
if (argc >= 5) {
|
if (argc >= 5) {
|
||||||
auto type = ox_atoi(args[2]);
|
auto type = ox_atoi(args[2]);
|
||||||
cout << args[3] << endl;
|
|
||||||
auto size = bytes(args[3]);
|
auto size = bytes(args[3]);
|
||||||
auto path = args[4];
|
auto path = args[4];
|
||||||
auto buff = (uint8_t*) malloc(size);
|
auto buff = new uint8_t[size];
|
||||||
|
|
||||||
cout << "Size: " << size << " bytes\n";
|
|
||||||
cout << "Type: " << type << endl;
|
|
||||||
|
|
||||||
if (size < sizeof(FileStore64)) {
|
if (size < sizeof(FileStore64)) {
|
||||||
err = 1;
|
err = 1;
|
||||||
@@ -126,7 +125,7 @@ int format(int argc, char **args) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
free(buff);
|
delete []buff;
|
||||||
|
|
||||||
if (err == 0) {
|
if (err == 0) {
|
||||||
fprintf(stderr, "Created file system %s\n", path);
|
fprintf(stderr, "Created file system %s\n", path);
|
||||||
@@ -149,7 +148,7 @@ int read(int argc, char **args) {
|
|||||||
auto fsBuff = loadFileBuff(fsPath, &fsSize);
|
auto fsBuff = loadFileBuff(fsPath, &fsSize);
|
||||||
|
|
||||||
if (fsBuff) {
|
if (fsBuff) {
|
||||||
auto fs = createFileSystem(fsBuff);
|
auto fs = createFileSystem(fsBuff, fsSize);
|
||||||
|
|
||||||
if (fs) {
|
if (fs) {
|
||||||
auto output = fs->read(inode, &fileSize);
|
auto output = fs->read(inode, &fileSize);
|
||||||
@@ -161,7 +160,7 @@ int read(int argc, char **args) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
delete fs;
|
delete fs;
|
||||||
free(fsBuff);
|
delete []fsBuff;
|
||||||
} else {
|
} else {
|
||||||
fprintf(stderr, "Invalid file system type: %d.\n", *(uint32_t*) fsBuff);
|
fprintf(stderr, "Invalid file system type: %d.\n", *(uint32_t*) fsBuff);
|
||||||
}
|
}
|
||||||
@@ -195,22 +194,22 @@ int write(int argc, char **args, bool expand) {
|
|||||||
if (itemsRead) {
|
if (itemsRead) {
|
||||||
auto srcBuff = loadFileBuff(srcPath, &srcSize);
|
auto srcBuff = loadFileBuff(srcPath, &srcSize);
|
||||||
if (srcBuff) {
|
if (srcBuff) {
|
||||||
auto fs = createFileSystem(fsBuff);
|
auto expanded = false;
|
||||||
|
auto fs = createFileSystem(fsBuff, fsSize);
|
||||||
if (fs) {
|
if (fs) {
|
||||||
if (expand && fs->available() <= srcSize) {
|
if (expand && fs->available() <= srcSize) {
|
||||||
auto needed = fs->spaceNeeded(inode, srcSize);
|
auto needed = fs->size() + fs->spaceNeeded(srcSize);
|
||||||
auto cloneBuff = new uint8_t[needed];
|
|
||||||
ox_memcpy(cloneBuff, fsBuff, fsSize);
|
|
||||||
|
|
||||||
delete fs;
|
|
||||||
delete []fsBuff;
|
|
||||||
|
|
||||||
fsBuff = cloneBuff;
|
|
||||||
fs = createFileSystem(fsBuff);
|
|
||||||
fsSize = needed;
|
fsSize = needed;
|
||||||
fs->resize(fsSize);
|
fs = expandCopyCleanup(fs, needed);
|
||||||
|
fsBuff = fs->buff();
|
||||||
}
|
}
|
||||||
err |= fs->write(inode, srcBuff, srcSize);
|
err |= fs->write(inode, srcBuff, srcSize);
|
||||||
|
|
||||||
|
// compact the file system if it was expanded
|
||||||
|
if (expanded) {
|
||||||
|
fs->resize();
|
||||||
|
}
|
||||||
|
|
||||||
if (err) {
|
if (err) {
|
||||||
fprintf(stderr, "Could not write to file system.\n");
|
fprintf(stderr, "Could not write to file system.\n");
|
||||||
}
|
}
|
||||||
@@ -233,7 +232,7 @@ int write(int argc, char **args, bool expand) {
|
|||||||
err = 1;
|
err = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
free(srcBuff);
|
delete []srcBuff;
|
||||||
} else {
|
} else {
|
||||||
err = 1;
|
err = 1;
|
||||||
fprintf(stderr, "Could not load source file: %s.\n", srcPath);
|
fprintf(stderr, "Could not load source file: %s.\n", srcPath);
|
||||||
@@ -258,7 +257,7 @@ int compact(int argc, char **args) {
|
|||||||
|
|
||||||
auto fsBuff = loadFileBuff(fsPath, &fsSize);
|
auto fsBuff = loadFileBuff(fsPath, &fsSize);
|
||||||
if (fsBuff) {
|
if (fsBuff) {
|
||||||
auto fs = createFileSystem(fsBuff);
|
auto fs = createFileSystem(fsBuff, fsSize);
|
||||||
|
|
||||||
if (fs) {
|
if (fs) {
|
||||||
fs->resize();
|
fs->resize();
|
||||||
@@ -279,7 +278,7 @@ int compact(int argc, char **args) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
delete fs;
|
delete fs;
|
||||||
free(fsBuff);
|
delete []fsBuff;
|
||||||
} else {
|
} else {
|
||||||
fprintf(stderr, "Could not open file: %s\n", fsPath);
|
fprintf(stderr, "Could not open file: %s\n", fsPath);
|
||||||
}
|
}
|
||||||
@@ -298,7 +297,7 @@ int remove(int argc, char **args) {
|
|||||||
|
|
||||||
auto fsBuff = loadFileBuff(fsPath, &fsSize);
|
auto fsBuff = loadFileBuff(fsPath, &fsSize);
|
||||||
if (fsBuff) {
|
if (fsBuff) {
|
||||||
auto fs = createFileSystem(fsBuff);
|
auto fs = createFileSystem(fsBuff, fsSize);
|
||||||
|
|
||||||
if (fs) {
|
if (fs) {
|
||||||
err = fs->remove(inode);
|
err = fs->remove(inode);
|
||||||
@@ -322,7 +321,7 @@ int remove(int argc, char **args) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
delete fs;
|
delete fs;
|
||||||
free(fsBuff);
|
delete []fsBuff;
|
||||||
} else {
|
} else {
|
||||||
fprintf(stderr, "Could not open file: %s\n", fsPath);
|
fprintf(stderr, "Could not open file: %s\n", fsPath);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@
|
|||||||
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
||||||
*/
|
*/
|
||||||
#include <map>
|
#include <map>
|
||||||
|
#include <string>
|
||||||
#include <ox/std/std.hpp>
|
#include <ox/std/std.hpp>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|||||||
@@ -23,11 +23,9 @@ typedef unsigned long uint64_t;
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
namespace ox {
|
namespace ox {
|
||||||
namespace std {
|
|
||||||
|
|
||||||
typedef uint32_t Error;
|
typedef uint32_t Error;
|
||||||
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(_LP64) || defined(__ppc64__) || defined(__aarch64__)
|
#if defined(_LP64) || defined(__ppc64__) || defined(__aarch64__)
|
||||||
|
|||||||
Reference in New Issue
Block a user