ОС обнаруживает make-файл
Я обычно работаю на нескольких разных компьютерах и нескольких разных операционных системах, таких как Mac OS X, Linux или Solaris. Для проекта, над которым я работаю, я извлекаю свой код из удаленного репозитория git.
Мне нравится работать над своими проектами независимо от того, в каком терминале я нахожусь. До сих пор я нашел способы обойти изменения ОС, меняя make-файл каждый раз, когда я переключаю компьютеры. Тем не менее, это утомительно и вызывает кучу головных болей.
Как я могу изменить мой make-файл, чтобы он определял, какую ОС я использую, и соответственно изменял синтаксис?
Вот make-файл:
cc = gcc -g
CC = g++ -g
yacc=$(YACC)
lex=$(FLEX)
all: assembler
assembler: y.tab.o lex.yy.o
$(CC) -o assembler y.tab.o lex.yy.o -ll -l y
assembler.o: assembler.c
$(cc) -o assembler.o assembler.c
y.tab.o: assem.y
$(yacc) -d assem.y
$(CC) -c y.tab.c
lex.yy.o: assem.l
$(lex) assem.l
$(cc) -c lex.yy.c
clean:
rm -f lex.yy.c y.tab.c y.tab.h assembler *.o *.tmp *.debug *.acts
14 ответов
Здесь уже есть много хороших ответов, но я хотел бы поделиться более полным примером, который оба:
- не предполагает
uname
существует в Windows - также обнаруживает процессор
Определенные здесь CCFLAGS не обязательно рекомендуются или идеальны; это именно то, что использовал проект, к которому я добавлял автоопределение ОС / ЦП.
ifeq ($(OS),Windows_NT)
CCFLAGS += -D WIN32
ifeq ($(PROCESSOR_ARCHITEW6432),AMD64)
CCFLAGS += -D AMD64
else
ifeq ($(PROCESSOR_ARCHITECTURE),AMD64)
CCFLAGS += -D AMD64
endif
ifeq ($(PROCESSOR_ARCHITECTURE),x86)
CCFLAGS += -D IA32
endif
endif
else
UNAME_S := $(shell uname -s)
ifeq ($(UNAME_S),Linux)
CCFLAGS += -D LINUX
endif
ifeq ($(UNAME_S),Darwin)
CCFLAGS += -D OSX
endif
UNAME_P := $(shell uname -p)
ifeq ($(UNAME_P),x86_64)
CCFLAGS += -D AMD64
endif
ifneq ($(filter %86,$(UNAME_P)),)
CCFLAGS += -D IA32
endif
ifneq ($(filter arm%,$(UNAME_P)),)
CCFLAGS += -D ARM
endif
endif
Команда uname ( http://developer.apple.com/documentation/Darwin/Reference/ManPages/man1/uname.1.html) без параметров должна сообщать вам имя операционной системы. Я бы использовал это, затем сделал бы условия на основе возвращаемого значения.
пример
UNAME := $(shell uname)
ifeq ($(UNAME), Linux)
# do something Linux-y
endif
ifeq ($(UNAME), Solaris)
# do something Solaris-y
endif
Определите операционную систему с помощью двух простых приемов:
- Сначала переменная среды
OS
- Тогда
uname
команда
ifeq ($(OS),Windows_NT) # is Windows_NT on XP, 2000, 7, Vista, 10...
detected_OS := Windows
else
detected_OS := $(shell uname) # same as "uname -s"
endif
Или более безопасный способ, если не на Windows и uname
недоступен:
ifeq ($(OS),Windows_NT)
detected_OS := Windows
else
detected_OS := $(shell sh -c 'uname 2>/dev/null || echo Unknown')
endif
Кен Джексон предлагает интересную альтернативу, если вы хотите отличить Cygwin/MinGW/MSYS/Windows. Посмотрите его ответ, который выглядит так:
ifeq '$(findstring ;,$(PATH))' ';'
detected_OS := Windows
else
detected_OS := $(shell uname 2>/dev/null || echo Unknown)
detected_OS := $(patsubst CYGWIN%,Cygwin,$(detected_OS))
detected_OS := $(patsubst MSYS%,MSYS,$(detected_OS))
detected_OS := $(patsubst MINGW%,MSYS,$(detected_OS))
endif
Затем вы можете выбрать соответствующий материал в зависимости от detected_OS
:
ifeq ($(detected_OS),Windows)
CFLAGS += -D WIN32
endif
ifeq ($(detected_OS),Darwin) # Mac OS X
CFLAGS += -D OSX
endif
ifeq ($(detected_OS),Linux)
CFLAGS += -D LINUX
endif
ifeq ($(detected_OS),GNU) # Debian GNU Hurd
CFLAGS += -D GNU_HURD
endif
ifeq ($(detected_OS),GNU/kFreeBSD) # Debian kFreeBSD
CFLAGS += -D GNU_kFreeBSD
endif
ifeq ($(detected_OS),FreeBSD)
CFLAGS += -D FreeBSD
endif
ifeq ($(detected_OS),NetBSD)
CFLAGS += -D NetBSD
endif
ifeq ($(detected_OS),DragonFly)
CFLAGS += -D DragonFly
endif
ifeq ($(detected_OS),Haiku)
CFLAGS += -D Haiku
endif
Заметки:
команда
uname
такой же какuname -s
потому что вариант-s
(--kernel-name
) по умолчанию. Понятно почемуuname -s
лучше, чемuname -o
,Использование
OS
(вместоuname
) упрощает алгоритм идентификации. Вы все еще можете использовать исключительноuname
, но вы должны иметь дело сif/else
блоки для проверки всех вариантов MinGW, Cygwin и др.Переменная среды
OS
всегда установлен на"Windows_NT"
на разных версиях Windows (см.%OS%
переменная окружения в Википедии).Альтернатива
OS
переменная средыMSVC
(он проверяет наличие MS Visual Studio, см. пример с использованием Visual C++).
Ниже я приведу полный пример использования make
а также gcc
построить общую библиотеку: *.so
или же *.dll
в зависимости от платформы. Пример настолько прост, насколько это возможно, чтобы быть более понятным.
Установить make
а также gcc
на Windows см. Cygwin или MinGW.
Мой пример основан на пяти файлах
├── lib
│ └── Makefile
│ └── hello.h
│ └── hello.c
└── app
└── Makefile
└── main.c
Напоминание: Makefile
отступ с помощью табуляции. Осторожно при копировании-вставке ниже примеров файлов.
Два Makefile
файлы
1. lib/Makefile
ifeq ($(OS),Windows_NT)
uname_S := Windows
else
uname_S := $(shell uname -s)
endif
ifeq ($(uname_S), Windows)
target = hello.dll
endif
ifeq ($(uname_S), Linux)
target = libhello.so
endif
#ifeq ($(uname_S), .....) #See https://stackru.com/a/27776822/938111
# target = .....
#endif
%.o: %.c
gcc -c $< -fPIC -o $@
# -c $< => $< is first file after ':' => Compile hello.c
# -fPIC => Position-Independent Code (required for shared lib)
# -o $@ => $@ is the target => Output file (-o) is hello.o
$(target): hello.o
gcc $^ -shared -o $@
# $^ => $^ expand to all prerequisites (after ':') => hello.o
# -shared => Generate shared library
# -o $@ => Output file (-o) is $@ (libhello.so or hello.dll)
2. app/Makefile
ifeq ($(OS),Windows_NT)
uname_S := Windows
else
uname_S := $(shell uname -s)
endif
ifeq ($(uname_S), Windows)
target = app.exe
endif
ifeq ($(uname_S), Linux)
target = app
endif
#ifeq ($(uname_S), .....) #See https://stackru.com/a/27776822/938111
# target = .....
#endif
%.o: %.c
gcc -c $< -I ../lib -o $@
# -c $< => compile (-c) $< (first file after :) = main.c
# -I ../lib => search headers (*.h) in directory ../lib
# -o $@ => output file (-o) is $@ (target) = main.o
$(target): main.o
gcc $^ -L../lib -lhello -o $@
# $^ => $^ (all files after the :) = main.o (here only one file)
# -L../lib => look for libraries in directory ../lib
# -lhello => use shared library hello (libhello.so or hello.dll)
# -o $@ => output file (-o) is $@ (target) = "app.exe" or "app"
Чтобы узнать больше, прочитайте документацию по автоматическим переменным, cfi.
Исходный код
- lib/hello.h
#ifndef HELLO_H_
#define HELLO_H_
const char* hello();
#endif
- lib/hello.c
#include "hello.h"
const char* hello()
{
return "hello";
}
- app/main.c
#include "hello.h" //hello()
#include <stdio.h> //puts()
int main()
{
const char* str = hello();
puts(str);
}
Сборка
Исправить копировать-вставить Makefile
(заменить начальные пробелы на одну таблицу).
> sed 's/^ */\t/' -i */Makefile
make
Команда одинакова на обеих платформах. Данный вывод находится на Unix-подобных ОС:
> make -C lib
make: Entering directory '/tmp/lib'
gcc -c hello.c -fPIC -o hello.o
# -c hello.c => hello.c is first file after ':' => Compile hello.c
# -fPIC => Position-Independent Code (required for shared lib)
# -o hello.o => hello.o is the target => Output file (-o) is hello.o
gcc hello.o -shared -o libhello.so
# hello.o => hello.o is the first after ':' => Link hello.o
# -shared => Generate shared library
# -o libhello.so => Output file (-o) is libhello.so (libhello.so or hello.dll)
make: Leaving directory '/tmp/lib'
> make -C app
make: Entering directory '/tmp/app'
gcc -c main.c -I ../lib -o main.o
# -c main.c => compile (-c) main.c (first file after :) = main.cpp
# -I ../lib => search headers (*.h) in directory ../lib
# -o main.o => output file (-o) is main.o (target) = main.o
gcc main.o -L../lib -lhello -o app
# main.o => main.o (all files after the :) = main.o (here only one file)
# -L../lib => look for libraries in directory ../lib
# -lhello => use shared library hello (libhello.so or hello.dll)
# -o app => output file (-o) is app.exe (target) = "app.exe" or "app"
make: Leaving directory '/tmp/app'
Бег
Приложение должно знать, где находится общая библиотека.
В Windows простое решение - скопировать библиотеку, в которой находится приложение:
> cp -v lib/hello.dll app
`lib/hello.dll' -> `app/hello.dll'
В Unix-подобных ОС вы можете использовать LD_LIBRARY_PATH
переменная окружения:
> export LD_LIBRARY_PATH=lib
Запустите команду в Windows:
> app/app.exe
hello
Запустите команду в Unix-подобных ОС:
> app/app
hello
Я недавно экспериментировал, чтобы ответить на этот вопрос, который я задавал себе. Вот мои выводы:
Так как в Windows вы не можете быть уверены, что uname
команда доступна, вы можете использовать gcc -dumpmachine
, Это покажет цель компилятора.
Там может быть также проблема при использовании uname
если вы хотите сделать кросс-компиляцию.
Вот пример списка возможных результатов gcc -dumpmachine
:
- mingw32
- i686-рс-Cygwin
- x86_64-RedHat-линукс
Вы можете проверить результат в make-файле следующим образом:
SYS := $(shell gcc -dumpmachine)
ifneq (, $(findstring linux, $(SYS)))
# Do Linux things
else ifneq(, $(findstring mingw, $(SYS)))
# Do MinGW things
else ifneq(, $(findstring cygwin, $(SYS)))
# Do Cygwin things
else
# Do things for others
endif
Это хорошо сработало для меня, но я не уверен, что это надежный способ получить тип системы. По крайней мере, это надежно в отношении MinGW, и это все, что мне нужно, поскольку для этого не требуется uname
команда или пакет MSYS в Windows.
Подводить итоги, uname
дает вам систему, на которой вы собираете, и gcc -dumpmachine
дает вам систему, для которой вы собираете.
Make-файл git содержит множество примеров того, как обойтись без autoconf/automake, но все же работает на множестве платформ Unixy.
Обновление: теперь я считаю этот ответ устаревшим. Я разместил новое идеальное решение ниже.
Если ваш make-файл может быть запущен не на Cygwin Windows, uname
может быть недоступен Это неудобно, но это потенциальное решение. Вы должны проверить Cygwin сначала, чтобы исключить это, потому что у него есть WINDOWS в его PATH
переменная окружения тоже.
ifneq (,$(findstring /cygdrive/,$(PATH)))
UNAME := Cygwin
else
ifneq (,$(findstring WINDOWS,$(PATH)))
UNAME := Windows
else
UNAME := $(shell uname -s)
endif
endif
Вот простое решение, которое проверяет, находитесь ли вы в среде Windows или в стиле posix (Linux/Unix/Cygwin/Mac):
ifeq ($(shell echo "check_quotes"),"check_quotes")
WINDOWS := yes
else
WINDOWS := no
endif
Он использует тот факт, что эхо существует как в posix-подобных средах, так и в Windows, и что в Windows оболочка не фильтрует кавычки.
Я наконец нашел идеальное решение, которое решает эту проблему для меня.
ifeq '$(findstring ;,$(PATH))' ';'
UNAME := Windows
else
UNAME := $(shell uname 2>/dev/null || echo Unknown)
UNAME := $(patsubst CYGWIN%,Cygwin,$(UNAME))
UNAME := $(patsubst MSYS%,MSYS,$(UNAME))
UNAME := $(patsubst MINGW%,MSYS,$(UNAME))
endif
Переменная UNAME установлена в Linux, Cygwin, MSYS, Windows, FreeBSD, NetBSD (или предположительно Solaris, Darwin, OpenBSD, AIX, HP-UX) или Неизвестно. Затем его можно сравнить в оставшейся части Makefile, чтобы отделить любые чувствительные к ОС переменные и команды.
Ключ заключается в том, что Windows использует точки с запятой для разделения путей в переменной PATH, тогда как все остальные используют двоеточия. (Можно сделать каталог Linux с символом ';' в имени и добавить его в PATH, что может привести к поломке, но кто будет делать такие вещи?) Это, по-видимому, наименее рискованный способ обнаружения родной Windows, потому что он не нужен вызов оболочки Cygwin и MSYS PATH используют двоеточия, поэтому для них требуется uname.
Обратите внимание, что переменная среды ОС может использоваться для обнаружения Windows, но не для различия между Cygwin и собственной Windows. Тестирование на эхо кавычек работает, но требует вызова оболочки.
К сожалению, Cygwin добавляет некоторую информацию о версии к выводу uname, поэтому я добавил вызовы 'patsubst', чтобы изменить его на 'Cygwin'. Кроме того, uname для MSYS на самом деле имеет три возможных выхода, начиная с MSYS или MINGW, но я использую также patsubst, чтобы преобразовать все в просто MSYS.
Если важно различать нативные системы Windows с и без некоторого uname.exe в пути, эту строку можно использовать вместо простого назначения:
UNAME := $(shell uname 2>NUL || echo Windows)
Конечно, во всех случаях требуется GNU make или другой make, который поддерживает используемые функции.
Я столкнулся с этой проблемой сегодня, и мне она понадобилась в Solaris, поэтому здесь есть стандартный способ POSIX (что-то очень похожее).
#Detect OS
UNAME = `uname`
# Build based on OS name
DetectOS:
-@make $(UNAME)
# OS is Linux, use GCC
Linux: program.c
@SHELL_VARIABLE="-D_LINUX_STUFF_HERE_"
rm -f program
gcc $(SHELL_VARIABLE) -o program program.c
# OS is Solaris, use c99
SunOS: program.c
@SHELL_VARIABLE="-D_SOLARIS_STUFF_HERE_"
rm -f program
c99 $(SHELL_VARIABLE) -o program program.c
Обратите внимание, что Makefiles чрезвычайно чувствительны к пробелам. Вот пример Makefile, который запускает дополнительную команду в OS X и работает в OS X и Linux. В целом, однако, autoconf / automake - это путь к чему-то нетривиальному.
UNAME: = $ (shell uname -s) CPP = g ++ CPPFLAGS = -pthread -ansi -Wall -Werror -pedantic -O0 -g3 -I / nexopia / include LDFLAGS = -pthread -L / nexopia / lib -lboost_system HEADERS = data_structures.h http_client.h load.h lock.h search.h server.h thread.h utility.h OBJECTS = http_client.o load.o lock.o search.o server.o thread.o utility.o vor.o все: vor чистить: rm -f $(ОБЪЕКТЫ) vor vor: $ (ОБЪЕКТЫ) $(CPP) $(LDFLAGS) -o vor $(ОБЪЕКТЫ) Ифек ($(UNAME), Дарвин) # Установить расположение библиотеки Boost install_name_tool -change libboost_system.dylib /nexopia/lib/libboost_system.dylib vor ENDIF %.o: %.cpp $(HEADERS) Makefile $(CPP) $(CPPFLAGS) -c $
Другой способ сделать это - использовать скрипт "configure". Если вы уже используете его в своем make-файле, вы можете использовать комбинацию uname и sed, чтобы все получилось. Сначала в вашем скрипте выполните:
UNAME=uname
Затем, чтобы поместить это в ваш Makefile, начните с Makefile.in, который должен иметь что-то вроде
UNAME=@@UNAME@@
в этом.
Используйте следующую команду sed в вашем скрипте configure после UNAME=uname
немного.
sed -e "s|@@UNAME@@|$UNAME|" < Makefile.in > Makefile
Теперь ваш make-файл должен иметь UNAME
определяется по желанию. Если /elif/else заявления все, что осталось!
Альтернативный способ, о котором я не видел, чтобы кто-то говорил, — это использование встроенной переменной . Программа, используемая в качестве оболочки, берется из переменной
SHELL
. В системах MS-Windows это, скорее всего, исполняемый файл с
.exe
расширение (как
sh.exe
).
В этом случае следующий условный тест:
ifeq ($(suffix $(SHELL)),.exe)
# Windows system
else
# Non-Windows system
endif
Будет иметь тот же результат, что и использование переменной окружения
OS
:
ifeq ($(OS),Windows_NT)
# Windows system
else
# Non-Windows system
endif
Тем не менее, кажется, что последнее решение является наиболее популярным, поэтому я бы рекомендовал вам придерживаться его.
У меня был случай, когда мне пришлось обнаружить разницу между двумя версиями Fedora, чтобы настроить параметры командной строки для inkscape:
- в Fedora31 по умолчанию используется inkscape 1.0beta, который использует--export-file
- в Fedora < 31 значение inkscape по умолчанию - 0,92, в котором используется --export-pdf
Мой Makefile содержит следующее
# set VERSION_ID from /etc/os-release
$(eval $(shell grep VERSION_ID /etc/os-release))
# select the inkscape export syntax
ifeq ($(VERSION_ID),31)
EXPORT = export-file
else
EXPORT = export-pdf
endif
# rule to convert inkscape SVG (drawing) to PDF
%.pdf : %.svg
inkscape --export-area-drawing $< --$(EXPORT)=$@
Это работает, потому что /etc/os-release
содержит строку
VERSION_ID=<value>
поэтому команда оболочки в Makefile возвращает строку VERSION_ID=<value>
, то команда eval воздействует на это, чтобы установить переменную Makefile VERSION_ID
. Очевидно, это можно настроить для других ОС в зависимости от того, как хранятся метаданные. Обратите внимание, что в Fedora нет переменной среды по умолчанию, которая указывает версию ОС, иначе я бы ее использовал!