Уникальный идентификатор оборудования в Mac OS X
Разработка Mac OS X - это для меня довольно новое животное, и я в процессе переноса по некоторому программному обеспечению. Для лицензирования и регистрации программного обеспечения мне нужно создать какой-то идентификатор оборудования. Это не должно быть ничего фантастического; MAC-адрес Ethernet, серийный номер жесткого диска, серийный номер CPU, что-то в этом роде.
У меня это покрыто Windows, но у меня нет подсказки на Mac. Любая идея о том, что мне нужно делать, или где я могу найти информацию об этом, была бы замечательной!
Edit:
Для кого-то, кого это интересует, это код, который я закончил с помощью класса Qt QProcess:
QProcess proc;
QStringList args;
args << "-c" << "ioreg -rd1 -c IOPlatformExpertDevice | awk '/IOPlatformUUID/ { print $3; }'";
proc.start( "/bin/bash", args );
proc.waitForFinished();
QString uID = proc.readAll();
Примечание. Я использую С++.
Ответы
Ответ 1
Попробуйте выполнить команду Terminal:
ioreg -rd1 -c IOPlatformExpertDevice | awk '/IOPlatformUUID/ { split($0, line, "\""); printf("%s\n", line[4]); }'
От здесь
Вот эта команда, завернутая в Cocoa (которая, вероятно, может быть сделана немного чище):
NSArray * args = [NSArray arrayWithObjects:@"-rd1", @"-c", @"IOPlatformExpertDevice", @"|", @"grep", @"model", nil];
NSTask * task = [NSTask new];
[task setLaunchPath:@"/usr/sbin/ioreg"];
[task setArguments:args];
NSPipe * pipe = [NSPipe new];
[task setStandardOutput:pipe];
[task launch];
NSArray * args2 = [NSArray arrayWithObjects:@"/IOPlatformUUID/ { split($0, line, \"\\\"\"); printf(\"%s\\n\", line[4]); }", nil];
NSTask * task2 = [NSTask new];
[task2 setLaunchPath:@"/usr/bin/awk"];
[task2 setArguments:args2];
NSPipe * pipe2 = [NSPipe new];
[task2 setStandardInput:pipe];
[task2 setStandardOutput:pipe2];
NSFileHandle * fileHandle2 = [pipe2 fileHandleForReading];
[task2 launch];
NSData * data = [fileHandle2 readDataToEndOfFile];
NSString * uuid = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
Ответ 2
Для C/С++:
void get_platform_uuid(char * buf, int bufSize) {
io_registry_entry_t ioRegistryRoot = IORegistryEntryFromPath(kIOMasterPortDefault, "IOService:/");
CFStringRef uuidCf = (CFStringRef) IORegistryEntryCreateCFProperty(ioRegistryRoot, CFSTR(kIOPlatformUUIDKey), kCFAllocatorDefault, 0);
IOObjectRelease(ioRegistryRoot);
CFStringGetCString(uuidCf, buf, bufSize, kCFStringEncodingMacRoman);
CFRelease(uuidCf);
}
Ответ 3
Это было бы легче ответить, если бы вы сказали нам, какой язык вы использовали. Информация доступна без команд оболочки через систему SystemConfiguration, а также через IOKit, если вы хотите, чтобы ваши руки были грязными.
- (NSString*) getMACAddress: (BOOL)stripColons {
NSMutableString *macAddress = nil;
NSArray *allInterfaces = (NSArray*)SCNetworkInterfaceCopyAll();
NSEnumerator *interfaceWalker = [allInterfaces objectEnumerator];
SCNetworkInterfaceRef curInterface = nil;
while ( curInterface = (SCNetworkInterfaceRef)[interfaceWalker nextObject] ) {
if ( [(NSString*)SCNetworkInterfaceGetBSDName(curInterface) isEqualToString:@"en0"] ) {
macAddress = [[(NSString*)SCNetworkInterfaceGetHardwareAddressString(curInterface) mutableCopy] autorelease];
if ( stripColons == YES ) {
[macAddress replaceOccurrencesOfString: @":" withString: @"" options: NSLiteralSearch range: NSMakeRange(0, [macAddress length])];
}
break;
}
}
return [[macAddress copy] autorelease];
}
Ответ 4
Почему бы не попробовать gethostuuid()
? Здесь документация из Руководства по системным вызовам Mac OS X:
NAME:
gethostuuid -- return a unique identifier for the current machine
СИНТАКСИС:
#include <unistd.h>
int gethostuuid(uuid_t id, const struct timespec *wait);
Описание:
Функция gethostuuid() возвращает 16-байтовый uuid_t, указанный id, который однозначно идентифицирует текущий компьютер. Имейте в виду, что аппаратные идентификаторы, используемые gethostuuid() для генерации UUID, могут быть изменены.
Аргумент wait - это указатель на struct timespec, который указывает максимальное время ожидания результата. Установка полей tv_sec и tv_nsec в ноль означает бесконечное ожидание, пока оно не завершится.
ВОЗВРАЩАЕМЫЕ ЗНАЧЕНИЯ:
Функция gethostuuid() возвращает ноль при успехе или -1 при ошибке.
ОШИБКИ
Функции gethostuuid() терпят неудачу, если:
[EFAULT] wait points to memory that is not a valid part of the
process address space.
[EWOULDBLOCK] The wait timeout expired before the UUID could be
obtained.
Ответ 5
/*
g++ mac_uuid.cpp -framework CoreFoundation -lIOKit
*/
#include <iostream>
#include <IOKit/IOKitLib.h>
using namespace std;
void get_platform_uuid(char * buf, int bufSize)
{
io_registry_entry_t ioRegistryRoot = IORegistryEntryFromPath(kIOMasterPortDefault, "IOService:/");
CFStringRef uuidCf = (CFStringRef) IORegistryEntryCreateCFProperty(ioRegistryRoot, CFSTR(kIOPlatformUUIDKey), kCFAllocatorDefault, 0);
IOObjectRelease(ioRegistryRoot);
CFStringGetCString(uuidCf, buf, bufSize, kCFStringEncodingMacRoman);
CFRelease(uuidCf);
}
int main()
{
char buf[512] = "";
get_platform_uuid(buf, sizeof(buf));
cout << buf << endl;
}
Ответ 6
Продолжительность:
system_profiler | grep 'Serial Number (system)'
в терминале возвращает то, что, вероятно, является уникальным идентификатором. Это работает на моем ящике 10.5, я не уверен, какая правильная строка будет в других версиях OS X.
Ответ 7
Как уже указывали некоторые люди выше, вы можете использовать команду Terminal для получения идентификатора оборудования.
Я предполагаю, что вы хотите сделать это в коде, но я бы посмотрел класс NSTask в Cocoa. Он в основном позволяет запускать команды терминала внутри вашего приложения.
Этот код является примером использования NSTask в Cocoa. Он настраивает среду для выполнения команды "killall". Он передает ему аргумент "Finder".
Это эквивалент запуска "killall Finder" в командной строке, который, очевидно, уничтожит Finder.
NSTask *aTask = [[NSTask alloc] init];
NSMutableArray *args = [NSMutableArray array];
[aTask setLaunchPath: @"/usr/bin/killall"];
[args addObject:[@"/Applications/Finder" lastPathComponent]];
[aTask setArguments:args];
[aTask launch];
[aTask release];
Ответ 8
System Profiler (в Приложениях - Утилиты) содержит большую часть этого типа информации. Он имеет ваш серийный номер и ваш адрес mac (нет отношения к Mac. Все компьютеры имеют адрес mac, который очень уникален для каждой сетевой карты).