/* * Copyright (c) 2001, Robert Collins. * * This program is free software; you can redistribute it and/or 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 * (at your option) any later version. * * A copy of the GNU General Public License can be found at * http://www.gnu.org/ * * Written by Robert Collins * */ #if 0 static const char *cvsid = "\n%%% $Id: source.cc,v 2.5 2001/11/01 13:22:46 rbcollins Exp $\n"; #endif #include "win32.h" #include #include #include "log.h" #include "port.h" #include "mount.h" #include "mkdir.h" #include "mklink2.h" #include #include "io_stream.h" #include "io_stream_cygfile.h" /* For set mtime */ #define FACTOR (0x19db1ded53ea710LL) #define NSPERSEC 10000000LL static void get_root_dir_now () { if (get_root_dir ()) return; read_mounts (); } io_stream_cygfile::io_stream_cygfile (const char *name, const char *mode) { fname = NULL; fp = NULL; errno = 0; if (!name || IsBadStringPtr (name, MAX_PATH) || !name[0] || !mode || IsBadStringPtr (mode, 5) || !mode[0]) return; /* do this every time because the mount points may change due to fwd/back button use... * TODO: make this less...manual */ get_root_dir_now (); if (!get_root_dir ()) /* TODO: assign a errno for "no mount table :} " */ return; fname = cygpath (name, 0); fp = fopen (fname, mode); if (!fp) lasterr = errno; } io_stream_cygfile::~io_stream_cygfile () { if (fname) free (fname); if (fp) fclose (fp); } /* Static members */ int io_stream_cygfile::exists (const char *path) { get_root_dir_now (); if (get_root_dir () && _access (cygpath (path, 0), 0) == 0) return 1; return 0; } int io_stream_cygfile::remove (const char *path) { if (!path) return 1; get_root_dir_now (); if (!get_root_dir ()) /* TODO: assign a errno for "no mount table :} " */ return 1; unsigned long w = GetFileAttributes (cygpath (path, 0)); if (w != 0xffffffff && w & FILE_ATTRIBUTE_DIRECTORY) { char *tmp = (char *) malloc (strlen (cygpath (path, 0)) + 10); int i = 0; do { i++; sprintf (tmp, "%s.old-%d", cygpath (path, 0), i); } while (GetFileAttributes (tmp) != 0xffffffff); fprintf (stderr, "warning: moving directory \"%s\" out of the way.\n", path); MoveFile (cygpath (path, 0), tmp); free (tmp); } return !DeleteFileA (cygpath (path, 0)); } int io_stream_cygfile::mklink (const char *from, const char *to, io_stream_link_t linktype) { /* FIXME: badstring check */ if (!from || !to) return 1; switch (linktype) { case IO_STREAM_SYMLINK: return mkcygsymlink (cygpath (from, 0), to); case IO_STREAM_HARDLINK: return 1; } return 1; } /* virtuals */ ssize_t io_stream_cygfile::read (void *buffer, size_t len) { if (fp) return fread (buffer, 1, len, fp); return 0; } ssize_t io_stream_cygfile::write (void *buffer, size_t len) { if (fp) return fwrite (buffer, 1, len, fp); return 0; } ssize_t io_stream_cygfile::peek (void *buffer, size_t len) { log (LOG_TIMESTAMP, "io_stream_cygfile::peek called"); if (fp) { int pos = ftell (fp); ssize_t rv = fread (buffer, 1, len, fp); fseek (fp, pos, SEEK_SET); return rv; } return 0; } long io_stream_cygfile::tell () { if (fp) { return ftell (fp); } return 0; } int io_stream_cygfile::error () { if (fp) return ferror (fp); return lasterr; } int cygmkdir_p (int isadir, const char *name) { if (!name || IsBadStringPtr (name, MAX_PATH) || !name[0]) return 1; get_root_dir_now (); if (!get_root_dir ()) /* TODO: assign a errno for "no mount table :} " */ return 1; return mkdir_p (isadir, cygpath (name, 0)); } int io_stream_cygfile::set_mtime (int mtime) { if (!fname) return 1; long long ftimev = mtime * NSPERSEC + FACTOR; FILETIME ftime; ftime.dwHighDateTime = ftimev >> 32; ftime.dwLowDateTime = ftimev; HANDLE h = CreateFileA (fname, GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL | FILE_FLAG_BACKUP_SEMANTICS, 0); if (h) { SetFileTime (h, 0, 0, &ftime); CloseHandle (h); return 0; } return 1; }