// Copyright (c) 2011 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef CHROME_INSTALLER_MINI_INSTALLER_MINI_STRING_H_ #define CHROME_INSTALLER_MINI_INSTALLER_MINI_STRING_H_ #ifndef COMPILE_ASSERT // COMPILE_ASSERT macro borrowed from basictypes.h template <bool> struct CompileAssert {}; #define COMPILE_ASSERT(expr, msg) \ typedef CompileAssert<(bool(expr))> msg[bool(expr) ? 1 : -1] #endif namespace mini_installer { // NOTE: Do not assume that these string functions support UTF encoding. // This is fine for the purposes of the mini_installer, but you have // been warned! // Formats a sequence of |bytes| as hex. The |str| buffer must have room for // at least 2*|size| + 1. bool HexEncode(const void* bytes, size_t size, wchar_t* str, size_t str_size); // Counts the number of characters in the string up to a maximum of // alloc_size. The highest return value from this function can therefore be // alloc_size - 1 since |alloc_size| includes the \0 terminator. size_t SafeStrLen(const wchar_t* str, size_t alloc_size); // Simple replacement for CRT string copy method that does not overflow. // Returns true if the source was copied successfully otherwise returns false. // Parameter src is assumed to be NULL terminated and the NULL character is // copied over to string dest. bool SafeStrCopy(wchar_t* dest, size_t dest_size, const wchar_t* src); // Simple replacement for CRT string copy method that does not overflow. // Returns true if the source was copied successfully otherwise returns false. // Parameter src is assumed to be NULL terminated and the NULL character is // copied over to string dest. If the return value is false, the |dest| // string should be the same as it was before. bool SafeStrCat(wchar_t* dest, size_t dest_size, const wchar_t* src); // Function to check if a string (specified by str) ends with another string // (specified by end_str). bool StrEndsWith(const wchar_t* str, const wchar_t* end_str); // Function to check if a string (specified by str) starts with another string // (specified by start_str). bool StrStartsWith(const wchar_t* str, const wchar_t* start_str); // Case insensitive search of the first occurrence of |find| in |source|. const wchar_t* SearchStringI(const wchar_t* source, const wchar_t* find); // Searches for |tag| within |str|. Returns true if |tag| is found and is // immediately followed by '-' or is at the end of the string. If |position| // is non-NULL, the location of the tag is returned in |*position| on success. bool FindTagInStr(const wchar_t* str, const wchar_t* tag, const wchar_t** position); // Takes the path to file and returns a pointer to the filename component. For // example for input of c:\full\path\to\file.ext it returns pointer to file.ext. // It returns NULL if extension or path separator is not found. // |size| is the number of characters in |path| not including the string // terminator. wchar_t* GetNameFromPathExt(wchar_t* path, size_t size); // A string class that manages a fixed size buffer on the stack. // The methods in the class are based on the above string methods and the // class additionally is careful about proper buffer termination. template <size_t kCapacity> class StackString { public: StackString() { COMPILE_ASSERT(kCapacity != 0, invalid_buffer_size); buffer_[kCapacity] = L'\0'; // We always reserve 1 more than asked for. clear(); } // We do not expose a constructor that accepts a string pointer on purpose. // We expect the caller to call assign() and handle failures. // Returns the number of reserved characters in this buffer, _including_ // the reserved char for the terminator. size_t capacity() const { return kCapacity; } wchar_t* get() { return buffer_; } bool assign(const wchar_t* str) { return SafeStrCopy(buffer_, kCapacity, str); } bool append(const wchar_t* str) { return SafeStrCat(buffer_, kCapacity, str); } void clear() { buffer_[0] = L'\0'; } size_t length() const { return SafeStrLen(buffer_, kCapacity); } // Does a case insensitive search for a substring. const wchar_t* findi(const wchar_t* find) const { return SearchStringI(buffer_, find); } // Case insensitive string compare. int comparei(const wchar_t* str) const { return lstrcmpiW(buffer_, str); } // Case sensitive string compare. int compare(const wchar_t* str) const { return lstrcmpW(buffer_, str); } // Terminates the string at the specified location. // Note: this method has no effect if this object's length is less than // |location|. bool truncate_at(size_t location) { if (location >= kCapacity) return false; buffer_[location] = L'\0'; return true; } protected: // We reserve 1 more than what is asked for as a safeguard against // off-by-one errors. wchar_t buffer_[kCapacity + 1]; private: StackString(const StackString&); StackString& operator=(const StackString&); }; } // namespace mini_installer #endif // CHROME_INSTALLER_MINI_INSTALLER_MINI_STRING_H_