2013-07-10 00:45:36 +00:00
|
|
|
/*
|
|
|
|
* libjingle
|
|
|
|
* Copyright 2007 Google Inc.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
|
|
* and/or other materials provided with the distribution.
|
|
|
|
* 3. The name of the author may not be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* 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 AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
|
|
|
|
* EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
|
|
|
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 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, EVEN IF
|
|
|
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
#include "talk/base/common.h"
|
|
|
|
#include "talk/base/logging.h"
|
|
|
|
#include "talk/base/macutils.h"
|
|
|
|
#include "talk/base/scoped_ptr.h"
|
|
|
|
#include "talk/base/stringutils.h"
|
|
|
|
|
|
|
|
namespace talk_base {
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
bool ToUtf8(const CFStringRef str16, std::string* str8) {
|
2013-12-05 00:24:06 +00:00
|
|
|
if ((NULL == str16) || (NULL == str8)) {
|
2013-07-10 00:45:36 +00:00
|
|
|
return false;
|
2013-12-05 00:24:06 +00:00
|
|
|
}
|
2013-07-10 00:45:36 +00:00
|
|
|
size_t maxlen = CFStringGetMaximumSizeForEncoding(CFStringGetLength(str16),
|
2013-12-05 00:24:06 +00:00
|
|
|
kCFStringEncodingUTF8) + 1;
|
2013-10-25 21:18:33 +00:00
|
|
|
scoped_ptr<char[]> buffer(new char[maxlen]);
|
2013-07-10 00:45:36 +00:00
|
|
|
if (!buffer || !CFStringGetCString(str16, buffer.get(), maxlen,
|
2013-12-05 00:24:06 +00:00
|
|
|
kCFStringEncodingUTF8)) {
|
2013-07-10 00:45:36 +00:00
|
|
|
return false;
|
2013-12-05 00:24:06 +00:00
|
|
|
}
|
2013-07-10 00:45:36 +00:00
|
|
|
str8->assign(buffer.get());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ToUtf16(const std::string& str8, CFStringRef* str16) {
|
2013-12-05 00:24:06 +00:00
|
|
|
if (NULL == str16) {
|
2013-07-10 00:45:36 +00:00
|
|
|
return false;
|
2013-12-05 00:24:06 +00:00
|
|
|
}
|
2013-07-10 00:45:36 +00:00
|
|
|
*str16 = CFStringCreateWithBytes(kCFAllocatorDefault,
|
|
|
|
reinterpret_cast<const UInt8*>(str8.data()),
|
|
|
|
str8.length(), kCFStringEncodingUTF8,
|
|
|
|
false);
|
2013-12-05 00:24:06 +00:00
|
|
|
return NULL != *str16;
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef OSX
|
|
|
|
void DecodeFourChar(UInt32 fc, std::string* out) {
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << '\'';
|
|
|
|
bool printable = true;
|
|
|
|
for (int i = 3; i >= 0; --i) {
|
|
|
|
char ch = (fc >> (8 * i)) & 0xFF;
|
|
|
|
if (isprint(static_cast<unsigned char>(ch))) {
|
|
|
|
ss << ch;
|
|
|
|
} else {
|
|
|
|
printable = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (printable) {
|
|
|
|
ss << '\'';
|
|
|
|
} else {
|
|
|
|
ss.str("");
|
|
|
|
ss << "0x" << std::hex << fc;
|
|
|
|
}
|
|
|
|
out->append(ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool GetGestalt(OSType ostype, int* value) {
|
|
|
|
ASSERT(NULL != value);
|
|
|
|
SInt32 native_value;
|
|
|
|
OSStatus result = Gestalt(ostype, &native_value);
|
|
|
|
if (noErr == result) {
|
|
|
|
*value = native_value;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
std::string str;
|
|
|
|
DecodeFourChar(ostype, &str);
|
|
|
|
LOG_E(LS_ERROR, OS, result) << "Gestalt(" << str << ")";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GetOSVersion(int* major, int* minor, int* bugfix) {
|
|
|
|
ASSERT(major && minor && bugfix);
|
2013-12-05 00:24:06 +00:00
|
|
|
if (!GetGestalt(gestaltSystemVersion, major)) {
|
2013-07-10 00:45:36 +00:00
|
|
|
return false;
|
2013-12-05 00:24:06 +00:00
|
|
|
}
|
2013-07-10 00:45:36 +00:00
|
|
|
if (*major < 0x1040) {
|
|
|
|
*bugfix = *major & 0xF;
|
|
|
|
*minor = (*major >> 4) & 0xF;
|
|
|
|
*major = (*major >> 8);
|
|
|
|
return true;
|
|
|
|
}
|
2013-12-05 00:24:06 +00:00
|
|
|
return GetGestalt(gestaltSystemVersionMajor, major) &&
|
|
|
|
GetGestalt(gestaltSystemVersionMinor, minor) &&
|
|
|
|
GetGestalt(gestaltSystemVersionBugFix, bugfix);
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MacOSVersionName GetOSVersionName() {
|
|
|
|
int major = 0, minor = 0, bugfix = 0;
|
2013-12-05 00:24:06 +00:00
|
|
|
if (!GetOSVersion(&major, &minor, &bugfix)) {
|
2013-07-10 00:45:36 +00:00
|
|
|
return kMacOSUnknown;
|
2013-12-05 00:24:06 +00:00
|
|
|
}
|
2013-07-10 00:45:36 +00:00
|
|
|
if (major > 10) {
|
|
|
|
return kMacOSNewer;
|
|
|
|
}
|
|
|
|
if ((major < 10) || (minor < 3)) {
|
|
|
|
return kMacOSOlder;
|
|
|
|
}
|
|
|
|
switch (minor) {
|
|
|
|
case 3:
|
|
|
|
return kMacOSPanther;
|
|
|
|
case 4:
|
|
|
|
return kMacOSTiger;
|
|
|
|
case 5:
|
|
|
|
return kMacOSLeopard;
|
|
|
|
case 6:
|
|
|
|
return kMacOSSnowLeopard;
|
|
|
|
case 7:
|
|
|
|
return kMacOSLion;
|
|
|
|
case 8:
|
|
|
|
return kMacOSMountainLion;
|
2013-12-05 00:24:06 +00:00
|
|
|
case 9:
|
|
|
|
return kMacOSMavericks;
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
return kMacOSNewer;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GetQuickTimeVersion(std::string* out) {
|
|
|
|
int ver;
|
2013-12-05 00:24:06 +00:00
|
|
|
if (!GetGestalt(gestaltQuickTimeVersion, &ver)) {
|
2013-07-10 00:45:36 +00:00
|
|
|
return false;
|
2013-12-05 00:24:06 +00:00
|
|
|
}
|
2013-07-10 00:45:36 +00:00
|
|
|
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << std::hex << ver;
|
|
|
|
*out = ss.str();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RunAppleScript(const std::string& script) {
|
|
|
|
// TODO(thaloun): Add a .mm file that contains something like this:
|
|
|
|
// NSString source from script
|
|
|
|
// NSAppleScript* appleScript = [[NSAppleScript alloc] initWithSource:&source]
|
|
|
|
// if (appleScript != nil) {
|
|
|
|
// [appleScript executeAndReturnError:nil]
|
|
|
|
// [appleScript release]
|
|
|
|
#ifndef CARBON_DEPRECATED
|
|
|
|
ComponentInstance component = NULL;
|
|
|
|
AEDesc script_desc;
|
|
|
|
AEDesc result_data;
|
|
|
|
OSStatus err;
|
|
|
|
OSAID script_id, result_id;
|
|
|
|
|
|
|
|
AECreateDesc(typeNull, NULL, 0, &script_desc);
|
|
|
|
AECreateDesc(typeNull, NULL, 0, &result_data);
|
|
|
|
script_id = kOSANullScript;
|
|
|
|
result_id = kOSANullScript;
|
|
|
|
|
|
|
|
component = OpenDefaultComponent(kOSAComponentType, typeAppleScript);
|
|
|
|
if (component == NULL) {
|
|
|
|
LOG(LS_ERROR) << "Failed opening Apple Script component";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
err = AECreateDesc(typeUTF8Text, script.data(), script.size(), &script_desc);
|
|
|
|
if (err != noErr) {
|
|
|
|
CloseComponent(component);
|
|
|
|
LOG(LS_ERROR) << "Failed creating Apple Script description";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = OSACompile(component, &script_desc, kOSAModeCanInteract, &script_id);
|
|
|
|
if (err != noErr) {
|
|
|
|
AEDisposeDesc(&script_desc);
|
|
|
|
if (script_id != kOSANullScript) {
|
|
|
|
OSADispose(component, script_id);
|
|
|
|
}
|
|
|
|
CloseComponent(component);
|
|
|
|
LOG(LS_ERROR) << "Error compiling Apple Script";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = OSAExecute(component, script_id, kOSANullScript, kOSAModeCanInteract,
|
|
|
|
&result_id);
|
|
|
|
|
|
|
|
if (err == errOSAScriptError) {
|
|
|
|
LOG(LS_ERROR) << "Error when executing Apple Script: " << script;
|
|
|
|
AECreateDesc(typeNull, NULL, 0, &result_data);
|
|
|
|
OSAScriptError(component, kOSAErrorMessage, typeChar, &result_data);
|
|
|
|
int len = AEGetDescDataSize(&result_data);
|
|
|
|
char* data = (char*) malloc(len);
|
|
|
|
if (data != NULL) {
|
|
|
|
err = AEGetDescData(&result_data, data, len);
|
|
|
|
LOG(LS_ERROR) << "Script error: " << data;
|
|
|
|
}
|
|
|
|
AEDisposeDesc(&script_desc);
|
|
|
|
AEDisposeDesc(&result_data);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
AEDisposeDesc(&script_desc);
|
|
|
|
if (script_id != kOSANullScript) {
|
|
|
|
OSADispose(component, script_id);
|
|
|
|
}
|
|
|
|
if (result_id != kOSANullScript) {
|
|
|
|
OSADispose(component, result_id);
|
|
|
|
}
|
|
|
|
CloseComponent(component);
|
|
|
|
return true;
|
|
|
|
#else
|
|
|
|
// TODO(thaloun): Support applescripts with the NSAppleScript API.
|
|
|
|
return false;
|
|
|
|
#endif // CARBON_DEPRECATED
|
|
|
|
}
|
|
|
|
#endif // OSX
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
} // namespace talk_base
|