coreutils/src/sync/sync.rs

185 lines
6 KiB
Rust
Raw Normal View History

2014-07-06 08:13:36 +00:00
#![crate_name = "uusync"]
2014-06-15 14:25:00 +00:00
/*
* This file is part of the uutils coreutils package.
*
* (c) Alexander Fomin <xander.fomin@ya.ru>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
/* Last synced with: sync (GNU coreutils) 8.13 */
extern crate getopts;
extern crate libc;
use getopts::{optflag, getopts, usage};
2015-01-08 12:54:22 +00:00
#[path = "../common/util.rs"] #[macro_use] mod util;
2014-06-20 12:44:35 +00:00
2014-08-15 22:53:23 +00:00
static NAME: &'static str = "sync";
static VERSION: &'static str = "1.0.0";
2014-06-20 12:44:35 +00:00
#[cfg(unix)]
mod platform {
use super::libc;
extern {
fn sync() -> libc::c_void;
}
pub unsafe fn do_sync() -> int {
sync();
0
}
2014-06-15 14:25:00 +00:00
}
2014-06-20 12:44:35 +00:00
#[cfg(windows)]
mod platform {
pub use super::libc;
2014-09-24 05:28:07 +00:00
use std::{mem, string};
2014-08-15 22:53:23 +00:00
use std::ptr::null;
2014-06-20 12:44:35 +00:00
extern "system" {
fn CreateFileA(lpFileName: *const libc::c_char,
2014-06-20 12:44:35 +00:00
dwDesiredAccess: libc::uint32_t,
dwShareMode: libc::uint32_t,
lpSecurityAttributes: *const libc::c_void, // *LPSECURITY_ATTRIBUTES
2014-06-20 12:44:35 +00:00
dwCreationDisposition: libc::uint32_t,
dwFlagsAndAttributes: libc::uint32_t,
hTemplateFile: *const libc::c_void) -> *const libc::c_void;
fn GetDriveTypeA(lpRootPathName: *const libc::c_char) -> libc::c_uint;
2014-06-20 12:44:35 +00:00
fn GetLastError() -> libc::uint32_t;
fn FindFirstVolumeA(lpszVolumeName: *mut libc::c_char,
cchBufferLength: libc::uint32_t) -> *const libc::c_void;
fn FindNextVolumeA(hFindVolume: *const libc::c_void,
lpszVolumeName: *mut libc::c_char,
2014-06-20 12:44:35 +00:00
cchBufferLength: libc::uint32_t) -> libc::c_int;
fn FindVolumeClose(hFindVolume: *const libc::c_void) -> libc::c_int;
fn FlushFileBuffers(hFile: *const libc::c_void) -> libc::c_int;
2014-06-20 12:44:35 +00:00
}
#[allow(unused_unsafe)]
unsafe fn flush_volume(name: &str) {
2014-09-24 05:28:07 +00:00
let name_buffer = name.to_c_str().as_ptr();
if 0x00000003 == GetDriveTypeA(name_buffer) { // DRIVE_FIXED
let sliced_name = name.slice_to(name.len() - 1); // eliminate trailing backslash
let sliced_name_buffer = sliced_name.to_c_str().as_ptr();
match CreateFileA(sliced_name_buffer,
0xC0000000, // GENERIC_WRITE
0x00000003, // FILE_SHARE_WRITE,
null(),
0x00000003, // OPEN_EXISTING
0,
null()) {
_x if _x == -1 as *const libc::c_void => { // INVALID_HANDLE_VALUE
crash!(GetLastError(), "failed to create volume handle");
}
handle @ _ => {
if FlushFileBuffers(handle) == 0 {
crash!(GetLastError(), "failed to flush file buffer");
2014-06-20 12:44:35 +00:00
}
2014-09-24 05:28:07 +00:00
}
2014-06-20 12:44:35 +00:00
}
2014-09-24 05:28:07 +00:00
}
2014-06-20 12:44:35 +00:00
}
#[allow(unused_unsafe)]
unsafe fn find_first_volume() -> (String, *const libc::c_void) {
2014-08-15 22:53:23 +00:00
let mut name: [libc::c_char, ..260] = mem::uninitialized(); // MAX_PATH
match FindFirstVolumeA(name.as_mut_ptr(),
name.len() as libc::uint32_t) {
_x if _x == -1 as *const libc::c_void => { // INVALID_HANDLE_VALUE
2014-06-20 12:44:35 +00:00
crash!(GetLastError(), "failed to find first volume");
}
handle @ _ => {
2014-09-24 05:28:07 +00:00
(string::raw::from_buf(name.as_ptr() as *const u8), handle)
2014-06-20 12:44:35 +00:00
}
}
}
#[allow(unused_unsafe)]
unsafe fn find_all_volumes() -> Vec<String> {
match find_first_volume() {
(first_volume, next_volume_handle) => {
let mut volumes = Vec::from_elem(1, first_volume);
loop {
2014-08-15 22:53:23 +00:00
let mut name: [libc::c_char, ..260] = mem::uninitialized(); // MAX_PATH
match FindNextVolumeA(next_volume_handle,
name.as_mut_ptr(),
name.len() as libc::uint32_t) {
2014-06-20 12:44:35 +00:00
0 => {
match GetLastError() {
0x12 => { // ERROR_NO_MORE_FILES
FindVolumeClose(next_volume_handle); // ignore FindVolumeClose() failures
break;
}
err @ _ => {
crash!(err, "failed to find next volume");
}
}
}
_ => {
2014-09-24 05:28:07 +00:00
volumes.push(string::raw::from_buf(name.as_ptr() as *const u8));
2014-06-20 12:44:35 +00:00
}
}
}
volumes
}
}
}
pub unsafe fn do_sync() -> int {
let volumes = find_all_volumes();
for vol in volumes.iter() {
flush_volume(vol.as_slice());
}
0
}
2014-06-15 14:25:00 +00:00
}
pub fn uumain(args: Vec<String>) -> int {
2014-07-20 01:13:55 +00:00
let program = &args[0];
2014-06-15 14:25:00 +00:00
let options = [
optflag("h", "help", "display this help and exit"),
optflag("V", "version", "output version information and exit")
];
2014-11-19 20:55:25 +00:00
let matches = match getopts(args.tail(), &options) {
2014-06-15 14:25:00 +00:00
Ok(m) => { m }
2014-11-19 20:55:25 +00:00
_ => { help(program.as_slice(), &options); return 1 }
2014-06-15 14:25:00 +00:00
};
if matches.opt_present("h") {
2014-11-19 20:55:25 +00:00
help(program.as_slice(), &options);
2014-06-15 14:25:00 +00:00
return 0
}
if matches.opt_present("V") {
version();
return 0
}
2014-06-20 12:44:35 +00:00
uusync();
2014-06-15 14:25:00 +00:00
0
}
fn version() {
println!("{} (uutils) {}", NAME, VERSION);
2014-06-15 14:25:00 +00:00
println!("The MIT License");
println!("");
println!("Author -- Alexander Fomin.");
}
fn help(program: &str, options: &[getopts::OptGroup]) {
2014-11-21 09:09:43 +00:00
println!("Usage: {} [OPTION]", program);
print!("{}", usage("Force changed blocks to disk, update the super block.", options));
2014-06-15 14:25:00 +00:00
}
2014-06-20 12:44:35 +00:00
fn uusync() -> int {
unsafe {
platform::do_sync()
}
}