দুটি সহজ কৌশল ব্যবহার করে অপারেটিং সিস্টেমটি সনাক্ত করুন:
- প্রথমে পরিবেশ পরিবর্তনশীল
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
বা আরও নিরাপদ উপায়, যদি উইন্ডোজে নেই এবং uname
অনুপলব্ধ:
ifeq ($(OS),Windows_NT)
detected_OS := Windows
else
detected_OS := $(shell sh -c 'uname 2>/dev/null || echo Unknown')
endif
আপনি যদি সাইগউইন / মিনজিডাব্লু / এমএসওয়াইএস / উইন্ডোজকে আলাদা করতে চান তবে কেন জ্যাকসন একটি আকর্ষণীয় বিকল্প প্রস্তাব করেছেন। তার উত্তর দেখুন যা দেখতে দেখতে:
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
সমস্ত মিনিজিডাব্লু, সাইগউইন ইত্যাদির বিভিন্নতা পরীক্ষা করতে আপনাকে ব্লকগুলি মোকাবেলা করতে হবে।
পরিবেশের পরিবর্তনশীল OS
সর্বদা "Windows_NT"
বিভিন্ন উইন্ডোজ সংস্করণে সেট করা থাকে ( %OS%
উইকিপিডিয়ায় পরিবেশ পরিবর্তনশীল দেখুন )।
এর বিকল্প OS
হ'ল পরিবেশ পরিবর্তনশীল MSVC
(এটি এমএস ভিজ্যুয়াল স্টুডিওর উপস্থিতি পরীক্ষা করে , ভিজ্যুয়াল সি ++ ব্যবহার করে উদাহরণ দেখুন )।
নীচে আমি ব্যবহার করে make
এবং gcc
একটি ভাগ করা লাইব্রেরি তৈরি করতে একটি সম্পূর্ণ উদাহরণ সরবরাহ করি : *.so
বা *.dll
প্ল্যাটফর্মের উপর নির্ভর করে। আরও বোধগম্য হওয়ার উদাহরণটি যতটা সম্ভব সহজ।
উইন্ডোজ ইনস্টল করতে make
এবং সাইগউইন বা মিনজিডাব্লু দেখুন ।gcc
আমার উদাহরণ পাঁচটি ফাইলের উপর ভিত্তি করে
├── 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://stackoverflow.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://stackoverflow.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"
আরও জানতে, সিএফআই দ্বারা নির্দেশিত হিসাবে স্বয়ংক্রিয় চলক ডকুমেন্টেশন পড়ুন ।
উত্স কোড
- 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
কমান্ড উভয় প্ল্যাটফর্মের উপর একই। প্রদত্ত আউটপুটটি ইউনিক্স-এর মতো ওএসগুলিতে রয়েছে:
> 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'
রান
অ্যাপ্লিকেশনটি জানতে হবে যে ভাগ করা লাইব্রেরিটি কোথায়।
উইন্ডোজটিতে, একটি সহজ সমাধান হ'ল অ্যাপ্লিকেশনটি রয়েছে এমন লাইব্রেরিটি অনুলিপি করা:
> cp -v lib/hello.dll app
`lib/hello.dll' -> `app/hello.dll'
ইউনিক্সের মতো ওএসে, আপনি LD_LIBRARY_PATH
পরিবেশের পরিবর্তনশীলটি ব্যবহার করতে পারেন :
> export LD_LIBRARY_PATH=lib
উইন্ডোতে কমান্ডটি চালান:
> app/app.exe
hello
ইউনিক্সের মতো ওএসে কমান্ডটি চালান:
> app/app
hello
PROCESSOR_ARCHITECTURE
envvar প্রক্রিয়াটি 32-বিট বা 64-বিট কিনা তার উপর নির্ভর করে ভার্চুয়ালাইজড বলে মনে হচ্ছে। সুতরাং যদি আপনারmake
32-বিট হয় এবং আপনি একটি 64-বিট অ্যাপ্লিকেশন তৈরি করার চেষ্টা করছেন, এটি ব্যর্থ হবে। একযোগে এটা ব্যবহার করেPROCESSOR_ARCHITEW6432
আমার জন্য কাজ (দেখুন এই , এবং যে )