Audio plugin host https://kx.studio/carla
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

carla_shared.py 31KB

11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
10 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
10 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
7 years ago
7 years ago
10 years ago
7 years ago
7 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
9 years ago
9 years ago
11 years ago
10 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
11 years ago
10 years ago
10 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881
  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Common Carla code
  4. # Copyright (C) 2011-2021 Filipe Coelho <falktx@falktx.com>
  5. #
  6. # This program is free software; you can redistribute it and/or
  7. # modify it under the terms of the GNU General Public License as
  8. # published by the Free Software Foundation; either version 2 of
  9. # the License, or any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # For a full copy of the GNU General Public License see the doc/GPL.txt file.
  17. # ------------------------------------------------------------------------------------------------------------
  18. # Imports (Global)
  19. import os
  20. import sys
  21. from math import fmod
  22. # ------------------------------------------------------------------------------------------------------------
  23. # Imports (Signal)
  24. from signal import signal, SIGINT, SIGTERM
  25. try:
  26. from signal import SIGUSR1
  27. haveSIGUSR1 = True
  28. except:
  29. haveSIGUSR1 = False
  30. # ------------------------------------------------------------------------------------------------------------
  31. # Imports (PyQt5)
  32. from PyQt5.Qt import PYQT_VERSION_STR
  33. from PyQt5.QtCore import qFatal, QT_VERSION, QT_VERSION_STR, qWarning, QDir, QSettings
  34. from PyQt5.QtGui import QIcon
  35. from PyQt5.QtWidgets import QFileDialog, QMessageBox
  36. # ------------------------------------------------------------------------------------------------------------
  37. # Imports (Custom)
  38. from carla_backend import (
  39. kIs64bit, HAIKU, LINUX, MACOS, WINDOWS,
  40. MAX_DEFAULT_PARAMETERS,
  41. ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS,
  42. ENGINE_PROCESS_MODE_PATCHBAY,
  43. ENGINE_TRANSPORT_MODE_INTERNAL,
  44. ENGINE_TRANSPORT_MODE_JACK
  45. )
  46. # ------------------------------------------------------------------------------------------------------------
  47. # Config
  48. # These will be modified during install
  49. X_LIBDIR_X = None
  50. X_DATADIR_X = None
  51. # ------------------------------------------------------------------------------------------------------------
  52. # Platform specific stuff
  53. if WINDOWS:
  54. WINDIR = os.getenv("WINDIR")
  55. # ------------------------------------------------------------------------------------------------------------
  56. # Set Version
  57. VERSION = "2.3.0-alpha2"
  58. # ------------------------------------------------------------------------------------------------------------
  59. # Set TMP
  60. envTMP = os.getenv("TMP")
  61. if envTMP is None:
  62. if WINDOWS:
  63. qWarning("TMP variable not set")
  64. TMP = QDir.tempPath()
  65. else:
  66. TMP = envTMP
  67. if not os.path.exists(TMP):
  68. qWarning("TMP does not exist")
  69. TMP = "/"
  70. del envTMP
  71. # ------------------------------------------------------------------------------------------------------------
  72. # Set HOME
  73. envHOME = os.getenv("HOME")
  74. if envHOME is None:
  75. if LINUX or MACOS:
  76. qWarning("HOME variable not set")
  77. HOME = QDir.toNativeSeparators(QDir.homePath())
  78. else:
  79. HOME = envHOME
  80. if not os.path.exists(HOME):
  81. qWarning("HOME does not exist")
  82. HOME = TMP
  83. del envHOME
  84. # ------------------------------------------------------------------------------------------------------------
  85. # Set PATH
  86. envPATH = os.getenv("PATH")
  87. if envPATH is None:
  88. qWarning("PATH variable not set")
  89. if MACOS:
  90. PATH = ("/opt/local/bin", "/usr/local/bin", "/usr/bin", "/bin")
  91. elif WINDOWS:
  92. PATH = (os.path.join(WINDIR, "system32"), WINDIR)
  93. else:
  94. PATH = ("/usr/local/bin", "/usr/bin", "/bin")
  95. else:
  96. PATH = envPATH.split(os.pathsep)
  97. del envPATH
  98. # ------------------------------------------------------------------------------------------------------------
  99. # Static MIDI CC list
  100. MIDI_CC_LIST = (
  101. "01 [0x01] Modulation",
  102. "02 [0x02] Breath",
  103. "04 [0x04] Foot",
  104. "05 [0x05] Portamento",
  105. "07 [0x07] Volume",
  106. "08 [0x08] Balance",
  107. "10 [0x0A] Pan",
  108. "11 [0x0B] Expression",
  109. "12 [0x0C] FX Control 1",
  110. "13 [0x0D] FX Control 2",
  111. "16 [0x10] General Purpose 1",
  112. "17 [0x11] General Purpose 2",
  113. "18 [0x12] General Purpose 3",
  114. "19 [0x13] General Purpose 4",
  115. "70 [0x46] Control 1 [Variation]",
  116. "71 [0x47] Control 2 [Timbre]",
  117. "72 [0x48] Control 3 [Release]",
  118. "73 [0x49] Control 4 [Attack]",
  119. "74 [0x4A] Control 5 [Brightness]",
  120. "75 [0x4B] Control 6 [Decay]",
  121. "76 [0x4C] Control 7 [Vib Rate]",
  122. "77 [0x4D] Control 8 [Vib Depth]",
  123. "78 [0x4E] Control 9 [Vib Delay]",
  124. "79 [0x4F] Control 10 [Undefined]",
  125. "80 [0x50] General Purpose 5",
  126. "81 [0x51] General Purpose 6",
  127. "82 [0x52] General Purpose 7",
  128. "83 [0x53] General Purpose 8",
  129. "84 [0x54] Portamento Control",
  130. "91 [0x5B] FX 1 Depth [Reverb]",
  131. "92 [0x5C] FX 2 Depth [Tremolo]",
  132. "93 [0x5D] FX 3 Depth [Chorus]",
  133. "94 [0x5E] FX 4 Depth [Detune]",
  134. "95 [0x5F] FX 5 Depth [Phaser]"
  135. )
  136. MAX_MIDI_CC_LIST_ITEM = 95
  137. # ------------------------------------------------------------------------------------------------------------
  138. # PatchCanvas defines
  139. CANVAS_ANTIALIASING_SMALL = 1
  140. CANVAS_EYECANDY_SMALL = 1
  141. # ------------------------------------------------------------------------------------------------------------
  142. # Carla Settings keys
  143. CARLA_KEY_MAIN_PROJECT_FOLDER = "Main/ProjectFolder" # str
  144. CARLA_KEY_MAIN_USE_PRO_THEME = "Main/UseProTheme" # bool
  145. CARLA_KEY_MAIN_PRO_THEME_COLOR = "Main/ProThemeColor" # str
  146. CARLA_KEY_MAIN_REFRESH_INTERVAL = "Main/RefreshInterval" # int
  147. CARLA_KEY_MAIN_CONFIRM_EXIT = "Main/ConfirmExit" # bool
  148. CARLA_KEY_MAIN_SHOW_LOGS = "Main/ShowLogs" # bool
  149. CARLA_KEY_MAIN_SYSTEM_ICONS = "Main/SystemIcons" # bool
  150. CARLA_KEY_MAIN_EXPERIMENTAL = "Main/Experimental" # bool
  151. CARLA_KEY_CANVAS_THEME = "Canvas/Theme" # str
  152. CARLA_KEY_CANVAS_SIZE = "Canvas/Size" # str "NxN"
  153. CARLA_KEY_CANVAS_USE_BEZIER_LINES = "Canvas/UseBezierLines" # bool
  154. CARLA_KEY_CANVAS_AUTO_HIDE_GROUPS = "Canvas/AutoHideGroups" # bool
  155. CARLA_KEY_CANVAS_AUTO_SELECT_ITEMS = "Canvas/AutoSelectItems" # bool
  156. CARLA_KEY_CANVAS_EYE_CANDY = "Canvas/EyeCandy2" # bool
  157. CARLA_KEY_CANVAS_FANCY_EYE_CANDY = "Canvas/FancyEyeCandy" # bool
  158. CARLA_KEY_CANVAS_USE_OPENGL = "Canvas/UseOpenGL" # bool
  159. CARLA_KEY_CANVAS_ANTIALIASING = "Canvas/Antialiasing" # enum
  160. CARLA_KEY_CANVAS_HQ_ANTIALIASING = "Canvas/HQAntialiasing" # bool
  161. CARLA_KEY_CANVAS_INLINE_DISPLAYS = "Canvas/InlineDisplays" # bool
  162. CARLA_KEY_CANVAS_FULL_REPAINTS = "Canvas/FullRepaints" # bool
  163. CARLA_KEY_ENGINE_DRIVER_PREFIX = "Engine/Driver-"
  164. CARLA_KEY_ENGINE_AUDIO_DRIVER = "Engine/AudioDriver" # str
  165. CARLA_KEY_ENGINE_PROCESS_MODE = "Engine/ProcessMode" # enum
  166. CARLA_KEY_ENGINE_TRANSPORT_MODE = "Engine/TransportMode" # enum
  167. CARLA_KEY_ENGINE_TRANSPORT_EXTRA = "Engine/TransportExtra" # str
  168. CARLA_KEY_ENGINE_FORCE_STEREO = "Engine/ForceStereo" # bool
  169. CARLA_KEY_ENGINE_PREFER_PLUGIN_BRIDGES = "Engine/PreferPluginBridges" # bool
  170. CARLA_KEY_ENGINE_PREFER_UI_BRIDGES = "Engine/PreferUiBridges" # bool
  171. CARLA_KEY_ENGINE_MANAGE_UIS = "Engine/ManageUIs" # bool
  172. CARLA_KEY_ENGINE_UIS_ALWAYS_ON_TOP = "Engine/UIsAlwaysOnTop" # bool
  173. CARLA_KEY_ENGINE_MAX_PARAMETERS = "Engine/MaxParameters" # int
  174. CARLA_KEY_ENGINE_RESET_XRUNS = "Engine/ResetXruns" # bool
  175. CARLA_KEY_ENGINE_UI_BRIDGES_TIMEOUT = "Engine/UiBridgesTimeout" # int
  176. CARLA_KEY_OSC_ENABLED = "OSC/Enabled"
  177. CARLA_KEY_OSC_TCP_PORT_ENABLED = "OSC/TCPEnabled"
  178. CARLA_KEY_OSC_TCP_PORT_NUMBER = "OSC/TCPNumber"
  179. CARLA_KEY_OSC_TCP_PORT_RANDOM = "OSC/TCPRandom"
  180. CARLA_KEY_OSC_UDP_PORT_ENABLED = "OSC/UDPEnabled"
  181. CARLA_KEY_OSC_UDP_PORT_NUMBER = "OSC/UDPNumber"
  182. CARLA_KEY_OSC_UDP_PORT_RANDOM = "OSC/UDPRandom"
  183. CARLA_KEY_PATHS_AUDIO = "Paths/Audio"
  184. CARLA_KEY_PATHS_MIDI = "Paths/MIDI"
  185. CARLA_KEY_PATHS_LADSPA = "Paths/LADSPA"
  186. CARLA_KEY_PATHS_DSSI = "Paths/DSSI"
  187. CARLA_KEY_PATHS_LV2 = "Paths/LV2"
  188. CARLA_KEY_PATHS_VST2 = "Paths/VST2"
  189. CARLA_KEY_PATHS_VST3 = "Paths/VST3"
  190. CARLA_KEY_PATHS_SF2 = "Paths/SF2"
  191. CARLA_KEY_PATHS_SFZ = "Paths/SFZ"
  192. CARLA_KEY_WINE_EXECUTABLE = "Wine/Executable" # str
  193. CARLA_KEY_WINE_AUTO_PREFIX = "Wine/AutoPrefix" # bool
  194. CARLA_KEY_WINE_FALLBACK_PREFIX = "Wine/FallbackPrefix" # str
  195. CARLA_KEY_WINE_RT_PRIO_ENABLED = "Wine/RtPrioEnabled" # bool
  196. CARLA_KEY_WINE_BASE_RT_PRIO = "Wine/BaseRtPrio" # int
  197. CARLA_KEY_WINE_SERVER_RT_PRIO = "Wine/ServerRtPrio" # int
  198. CARLA_KEY_EXPERIMENTAL_PLUGIN_BRIDGES = "Experimental/PluginBridges" # bool
  199. CARLA_KEY_EXPERIMENTAL_WINE_BRIDGES = "Experimental/WineBridges" # bool
  200. CARLA_KEY_EXPERIMENTAL_JACK_APPS = "Experimental/JackApplications" # bool
  201. CARLA_KEY_EXPERIMENTAL_EXPORT_LV2 = "Experimental/ExportLV2" # bool
  202. CARLA_KEY_EXPERIMENTAL_PREVENT_BAD_BEHAVIOUR = "Experimental/PreventBadBehaviour" # bool
  203. CARLA_KEY_EXPERIMENTAL_LOAD_LIB_GLOBAL = "Experimental/LoadLibGlobal" # bool
  204. # if pro theme is on and color is black
  205. CARLA_KEY_CUSTOM_PAINTING = "UseCustomPainting" # bool
  206. # ------------------------------------------------------------------------------------------------------------
  207. # Carla Settings defaults
  208. # Main
  209. CARLA_DEFAULT_MAIN_PROJECT_FOLDER = HOME
  210. CARLA_DEFAULT_MAIN_USE_PRO_THEME = True
  211. CARLA_DEFAULT_MAIN_PRO_THEME_COLOR = "Black"
  212. CARLA_DEFAULT_MAIN_REFRESH_INTERVAL = 20
  213. CARLA_DEFAULT_MAIN_CONFIRM_EXIT = True
  214. CARLA_DEFAULT_MAIN_SHOW_LOGS = bool(not WINDOWS)
  215. CARLA_DEFAULT_MAIN_SYSTEM_ICONS = False
  216. CARLA_DEFAULT_MAIN_EXPERIMENTAL = False
  217. # Canvas
  218. CARLA_DEFAULT_CANVAS_THEME = "Modern Dark"
  219. CARLA_DEFAULT_CANVAS_SIZE = "3100x2400"
  220. CARLA_DEFAULT_CANVAS_SIZE_WIDTH = 3100
  221. CARLA_DEFAULT_CANVAS_SIZE_HEIGHT = 2400
  222. CARLA_DEFAULT_CANVAS_USE_BEZIER_LINES = True
  223. CARLA_DEFAULT_CANVAS_AUTO_HIDE_GROUPS = True
  224. CARLA_DEFAULT_CANVAS_AUTO_SELECT_ITEMS = False
  225. CARLA_DEFAULT_CANVAS_EYE_CANDY = True
  226. CARLA_DEFAULT_CANVAS_FANCY_EYE_CANDY = False
  227. CARLA_DEFAULT_CANVAS_USE_OPENGL = False
  228. CARLA_DEFAULT_CANVAS_ANTIALIASING = CANVAS_ANTIALIASING_SMALL
  229. CARLA_DEFAULT_CANVAS_HQ_ANTIALIASING = False
  230. CARLA_DEFAULT_CANVAS_INLINE_DISPLAYS = False
  231. CARLA_DEFAULT_CANVAS_FULL_REPAINTS = False
  232. # Engine
  233. CARLA_DEFAULT_FORCE_STEREO = False
  234. CARLA_DEFAULT_PREFER_PLUGIN_BRIDGES = False
  235. CARLA_DEFAULT_PREFER_UI_BRIDGES = True
  236. CARLA_DEFAULT_MANAGE_UIS = True
  237. CARLA_DEFAULT_UIS_ALWAYS_ON_TOP = False
  238. CARLA_DEFAULT_MAX_PARAMETERS = MAX_DEFAULT_PARAMETERS
  239. CARLA_DEFAULT_RESET_XRUNS = False
  240. CARLA_DEFAULT_UI_BRIDGES_TIMEOUT = 4000
  241. CARLA_DEFAULT_AUDIO_BUFFER_SIZE = 512
  242. CARLA_DEFAULT_AUDIO_SAMPLE_RATE = 44100
  243. CARLA_DEFAULT_AUDIO_TRIPLE_BUFFER = False
  244. if WINDOWS:
  245. CARLA_DEFAULT_AUDIO_DRIVER = "DirectSound"
  246. elif MACOS:
  247. CARLA_DEFAULT_AUDIO_DRIVER = "CoreAudio"
  248. elif os.path.exists("/usr/bin/jackd") or os.path.exists("/usr/bin/jackdbus"):
  249. CARLA_DEFAULT_AUDIO_DRIVER = "JACK"
  250. else:
  251. CARLA_DEFAULT_AUDIO_DRIVER = "PulseAudio"
  252. if CARLA_DEFAULT_AUDIO_DRIVER == "JACK":
  253. CARLA_DEFAULT_PROCESS_MODE = ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS
  254. CARLA_DEFAULT_TRANSPORT_MODE = ENGINE_TRANSPORT_MODE_JACK
  255. else:
  256. CARLA_DEFAULT_PROCESS_MODE = ENGINE_PROCESS_MODE_PATCHBAY
  257. CARLA_DEFAULT_TRANSPORT_MODE = ENGINE_TRANSPORT_MODE_INTERNAL
  258. # OSC
  259. CARLA_DEFAULT_OSC_ENABLED = not (MACOS or WINDOWS)
  260. CARLA_DEFAULT_OSC_TCP_PORT_ENABLED = True
  261. CARLA_DEFAULT_OSC_TCP_PORT_NUMBER = 22752
  262. CARLA_DEFAULT_OSC_TCP_PORT_RANDOM = False
  263. CARLA_DEFAULT_OSC_UDP_PORT_ENABLED = True
  264. CARLA_DEFAULT_OSC_UDP_PORT_NUMBER = 22752
  265. CARLA_DEFAULT_OSC_UDP_PORT_RANDOM = False
  266. # Wine
  267. CARLA_DEFAULT_WINE_EXECUTABLE = "wine"
  268. CARLA_DEFAULT_WINE_AUTO_PREFIX = True
  269. CARLA_DEFAULT_WINE_FALLBACK_PREFIX = os.path.expanduser("~/.wine")
  270. CARLA_DEFAULT_WINE_RT_PRIO_ENABLED = True
  271. CARLA_DEFAULT_WINE_BASE_RT_PRIO = 15
  272. CARLA_DEFAULT_WINE_SERVER_RT_PRIO = 10
  273. # Experimental
  274. CARLA_DEFAULT_EXPERIMENTAL_PLUGIN_BRIDGES = False
  275. CARLA_DEFAULT_EXPERIMENTAL_WINE_BRIDGES = False
  276. CARLA_DEFAULT_EXPERIMENTAL_JACK_APPS = False
  277. CARLA_DEFAULT_EXPERIMENTAL_LV2_EXPORT = False
  278. CARLA_DEFAULT_EXPERIMENTAL_PREVENT_BAD_BEHAVIOUR = False
  279. CARLA_DEFAULT_EXPERIMENTAL_LOAD_LIB_GLOBAL = False
  280. # ------------------------------------------------------------------------------------------------------------
  281. # Default File Folders
  282. CARLA_DEFAULT_FILE_PATH_AUDIO = []
  283. CARLA_DEFAULT_FILE_PATH_MIDI = []
  284. # ------------------------------------------------------------------------------------------------------------
  285. # Default Plugin Folders (get)
  286. DEFAULT_LADSPA_PATH = ""
  287. DEFAULT_DSSI_PATH = ""
  288. DEFAULT_LV2_PATH = ""
  289. DEFAULT_VST2_PATH = ""
  290. DEFAULT_VST3_PATH = ""
  291. DEFAULT_SF2_PATH = ""
  292. DEFAULT_SFZ_PATH = ""
  293. if WINDOWS:
  294. splitter = ";"
  295. APPDATA = os.getenv("APPDATA")
  296. PROGRAMFILES = os.getenv("PROGRAMFILES")
  297. PROGRAMFILESx86 = os.getenv("PROGRAMFILES(x86)")
  298. COMMONPROGRAMFILES = os.getenv("COMMONPROGRAMFILES")
  299. COMMONPROGRAMFILESx86 = os.getenv("COMMONPROGRAMFILES(x86)")
  300. # Small integrity tests
  301. if not APPDATA:
  302. qFatal("APPDATA variable not set, cannot continue")
  303. sys.exit(1)
  304. if not PROGRAMFILES:
  305. qFatal("PROGRAMFILES variable not set, cannot continue")
  306. sys.exit(1)
  307. if not COMMONPROGRAMFILES:
  308. qFatal("COMMONPROGRAMFILES variable not set, cannot continue")
  309. sys.exit(1)
  310. DEFAULT_LADSPA_PATH = APPDATA + "\\LADSPA"
  311. DEFAULT_LADSPA_PATH += ";" + PROGRAMFILES + "\\LADSPA"
  312. DEFAULT_DSSI_PATH = APPDATA + "\\DSSI"
  313. DEFAULT_DSSI_PATH += ";" + PROGRAMFILES + "\\DSSI"
  314. DEFAULT_LV2_PATH = APPDATA + "\\LV2"
  315. DEFAULT_LV2_PATH += ";" + COMMONPROGRAMFILES + "\\LV2"
  316. DEFAULT_VST2_PATH = PROGRAMFILES + "\\VstPlugins"
  317. DEFAULT_VST2_PATH += ";" + PROGRAMFILES + "\\Steinberg\\VstPlugins"
  318. if kIs64bit:
  319. DEFAULT_VST2_PATH += ";" + COMMONPROGRAMFILES + "\\VST2"
  320. DEFAULT_VST3_PATH = COMMONPROGRAMFILES + "\\VST3"
  321. DEFAULT_SF2_PATH = APPDATA + "\\SF2"
  322. DEFAULT_SFZ_PATH = APPDATA + "\\SFZ"
  323. if PROGRAMFILESx86:
  324. DEFAULT_LADSPA_PATH += ";" + PROGRAMFILESx86 + "\\LADSPA"
  325. DEFAULT_DSSI_PATH += ";" + PROGRAMFILESx86 + "\\DSSI"
  326. DEFAULT_VST2_PATH += ";" + PROGRAMFILESx86 + "\\VstPlugins"
  327. DEFAULT_VST2_PATH += ";" + PROGRAMFILESx86 + "\\Steinberg\\VstPlugins"
  328. if COMMONPROGRAMFILESx86:
  329. DEFAULT_VST3_PATH += COMMONPROGRAMFILESx86 + "\\VST3"
  330. elif HAIKU:
  331. splitter = ":"
  332. DEFAULT_LADSPA_PATH = HOME + "/.ladspa"
  333. DEFAULT_LADSPA_PATH += ":/system/add-ons/media/ladspaplugins"
  334. DEFAULT_LADSPA_PATH += ":/system/lib/ladspa"
  335. DEFAULT_DSSI_PATH = HOME + "/.dssi"
  336. DEFAULT_DSSI_PATH += ":/system/add-ons/media/dssiplugins"
  337. DEFAULT_DSSI_PATH += ":/system/lib/dssi"
  338. DEFAULT_LV2_PATH = HOME + "/.lv2"
  339. DEFAULT_LV2_PATH += ":/system/add-ons/media/lv2plugins"
  340. DEFAULT_VST2_PATH = HOME + "/.vst"
  341. DEFAULT_VST2_PATH += ":/system/add-ons/media/vstplugins"
  342. DEFAULT_VST3_PATH = HOME + "/.vst3"
  343. DEFAULT_VST3_PATH += ":/system/add-ons/media/vst3plugins"
  344. elif MACOS:
  345. splitter = ":"
  346. DEFAULT_LADSPA_PATH = HOME + "/Library/Audio/Plug-Ins/LADSPA"
  347. DEFAULT_LADSPA_PATH += ":/Library/Audio/Plug-Ins/LADSPA"
  348. DEFAULT_DSSI_PATH = HOME + "/Library/Audio/Plug-Ins/DSSI"
  349. DEFAULT_DSSI_PATH += ":/Library/Audio/Plug-Ins/DSSI"
  350. DEFAULT_LV2_PATH = HOME + "/Library/Audio/Plug-Ins/LV2"
  351. DEFAULT_LV2_PATH += ":/Library/Audio/Plug-Ins/LV2"
  352. DEFAULT_VST2_PATH = HOME + "/Library/Audio/Plug-Ins/VST"
  353. DEFAULT_VST2_PATH += ":/Library/Audio/Plug-Ins/VST"
  354. DEFAULT_VST3_PATH = HOME + "/Library/Audio/Plug-Ins/VST3"
  355. DEFAULT_VST3_PATH += ":/Library/Audio/Plug-Ins/VST3"
  356. else:
  357. splitter = ":"
  358. DEFAULT_LADSPA_PATH = HOME + "/.ladspa"
  359. DEFAULT_LADSPA_PATH += ":/usr/lib/ladspa"
  360. DEFAULT_LADSPA_PATH += ":/usr/local/lib/ladspa"
  361. DEFAULT_DSSI_PATH = HOME + "/.dssi"
  362. DEFAULT_DSSI_PATH += ":/usr/lib/dssi"
  363. DEFAULT_DSSI_PATH += ":/usr/local/lib/dssi"
  364. DEFAULT_LV2_PATH = HOME + "/.lv2"
  365. DEFAULT_LV2_PATH += ":/usr/lib/lv2"
  366. DEFAULT_LV2_PATH += ":/usr/local/lib/lv2"
  367. DEFAULT_VST2_PATH = HOME + "/.vst"
  368. DEFAULT_VST2_PATH += ":/usr/lib/vst"
  369. DEFAULT_VST2_PATH += ":/usr/local/lib/vst"
  370. DEFAULT_VST2_PATH += HOME + "/.lxvst"
  371. DEFAULT_VST2_PATH += ":/usr/lib/lxvst"
  372. DEFAULT_VST2_PATH += ":/usr/local/lib/lxvst"
  373. DEFAULT_VST3_PATH = HOME + "/.vst3"
  374. DEFAULT_VST3_PATH += ":/usr/lib/vst3"
  375. DEFAULT_VST3_PATH += ":/usr/local/lib/vst3"
  376. DEFAULT_SF2_PATH = HOME + "/.sounds/sf2"
  377. DEFAULT_SF2_PATH += HOME + "/.sounds/sf3"
  378. DEFAULT_SF2_PATH += ":/usr/share/sounds/sf2"
  379. DEFAULT_SF2_PATH += ":/usr/share/sounds/sf3"
  380. DEFAULT_SF2_PATH += ":/usr/share/soundfonts"
  381. DEFAULT_SFZ_PATH = HOME + "/.sounds/sfz"
  382. DEFAULT_SFZ_PATH += ":/usr/share/sounds/sfz"
  383. if not WINDOWS:
  384. winePrefix = os.getenv("WINEPREFIX")
  385. if not winePrefix:
  386. winePrefix = HOME + "/.wine"
  387. if os.path.exists(winePrefix):
  388. DEFAULT_VST2_PATH += ":" + winePrefix + "/drive_c/Program Files/VstPlugins"
  389. DEFAULT_VST3_PATH += ":" + winePrefix + "/drive_c/Program Files/Common Files/VST3"
  390. if kIs64bit and os.path.exists(winePrefix + "/drive_c/Program Files (x86)"):
  391. DEFAULT_VST2_PATH += ":" + winePrefix + "/drive_c/Program Files (x86)/VstPlugins"
  392. DEFAULT_VST3_PATH += ":" + winePrefix + "/drive_c/Program Files (x86)/Common Files/VST3"
  393. del winePrefix
  394. # ------------------------------------------------------------------------------------------------------------
  395. # Default Plugin Folders (set)
  396. readEnvVars = True
  397. if WINDOWS:
  398. # Check if running Wine. If yes, ignore env vars
  399. # pylint: disable=import-error
  400. from winreg import ConnectRegistry, OpenKey, CloseKey, HKEY_CURRENT_USER
  401. # pylint: enable=import-error
  402. _reg = ConnectRegistry(None, HKEY_CURRENT_USER)
  403. try:
  404. _key = OpenKey(_reg, r"SOFTWARE\Wine")
  405. CloseKey(_key)
  406. del _key
  407. readEnvVars = False
  408. except:
  409. pass
  410. CloseKey(_reg)
  411. del _reg
  412. if readEnvVars:
  413. CARLA_DEFAULT_LADSPA_PATH = os.getenv("LADSPA_PATH", DEFAULT_LADSPA_PATH).split(splitter)
  414. CARLA_DEFAULT_DSSI_PATH = os.getenv("DSSI_PATH", DEFAULT_DSSI_PATH).split(splitter)
  415. CARLA_DEFAULT_LV2_PATH = os.getenv("LV2_PATH", DEFAULT_LV2_PATH).split(splitter)
  416. CARLA_DEFAULT_VST2_PATH = os.getenv("VST_PATH", DEFAULT_VST2_PATH).split(splitter)
  417. CARLA_DEFAULT_VST3_PATH = os.getenv("VST3_PATH", DEFAULT_VST3_PATH).split(splitter)
  418. CARLA_DEFAULT_SF2_PATH = os.getenv("SF2_PATH", DEFAULT_SF2_PATH).split(splitter)
  419. CARLA_DEFAULT_SFZ_PATH = os.getenv("SFZ_PATH", DEFAULT_SFZ_PATH).split(splitter)
  420. else:
  421. CARLA_DEFAULT_LADSPA_PATH = DEFAULT_LADSPA_PATH.split(splitter)
  422. CARLA_DEFAULT_DSSI_PATH = DEFAULT_DSSI_PATH.split(splitter)
  423. CARLA_DEFAULT_LV2_PATH = DEFAULT_LV2_PATH.split(splitter)
  424. CARLA_DEFAULT_VST2_PATH = DEFAULT_VST2_PATH.split(splitter)
  425. CARLA_DEFAULT_VST3_PATH = DEFAULT_VST3_PATH.split(splitter)
  426. CARLA_DEFAULT_SF2_PATH = DEFAULT_SF2_PATH.split(splitter)
  427. CARLA_DEFAULT_SFZ_PATH = DEFAULT_SFZ_PATH.split(splitter)
  428. # ------------------------------------------------------------------------------------------------------------
  429. # Default Plugin Folders (cleanup)
  430. del DEFAULT_LADSPA_PATH
  431. del DEFAULT_DSSI_PATH
  432. del DEFAULT_LV2_PATH
  433. del DEFAULT_VST2_PATH
  434. del DEFAULT_VST3_PATH
  435. del DEFAULT_SF2_PATH
  436. del DEFAULT_SFZ_PATH
  437. # ------------------------------------------------------------------------------------------------------------
  438. # Global Carla object
  439. class CarlaObject():
  440. def __init__(self):
  441. self.cnprefix = "" # Client name prefix
  442. self.gui = None # Host Window
  443. self.nogui = False # Skip UI
  444. self.term = False # Terminated by OS signal
  445. self.utils = None # Utils object
  446. gCarla = CarlaObject()
  447. # ------------------------------------------------------------------------------------------------------------
  448. # Set CWD
  449. CWD = sys.path[0]
  450. if not CWD:
  451. CWD = os.path.dirname(sys.argv[0])
  452. # make it work with cxfreeze
  453. if os.path.isfile(CWD):
  454. CWD = os.path.dirname(CWD)
  455. if CWD.endswith("/lib"):
  456. CWD = CWD.rsplit("/lib",1)[0]
  457. CXFREEZE = True
  458. else:
  459. CXFREEZE = False
  460. # ------------------------------------------------------------------------------------------------------------
  461. # Set DLL_EXTENSION
  462. if WINDOWS:
  463. DLL_EXTENSION = "dll"
  464. elif MACOS:
  465. DLL_EXTENSION = "dylib"
  466. else:
  467. DLL_EXTENSION = "so"
  468. # ------------------------------------------------------------------------------------------------------------
  469. # Find decimal points for a parameter, using step and stepSmall
  470. def countDecimalPoints(step, stepSmall):
  471. if stepSmall >= 1.0:
  472. return 0
  473. if step >= 1.0:
  474. return 2
  475. count = 0
  476. value = fmod(abs(stepSmall), 1)
  477. while 0.001 < value < 0.99 and count < 6:
  478. value = fmod(value*10, 1)
  479. count += 1
  480. return count
  481. # ------------------------------------------------------------------------------------------------------------
  482. # Check if a value is a number (float support)
  483. def isNumber(value):
  484. try:
  485. float(value)
  486. return True
  487. except:
  488. return False
  489. # ------------------------------------------------------------------------------------------------------------
  490. # Convert a value to a list
  491. def toList(value):
  492. if value is None:
  493. return []
  494. if not isinstance(value, list):
  495. return [value]
  496. return value
  497. # ------------------------------------------------------------------------------------------------------------
  498. # Get Icon from user theme, using our own as backup (Oxygen)
  499. def getIcon(icon, size, qrcformat):
  500. return QIcon.fromTheme(icon, QIcon(":/%ix%i/%s.%s" % (size, size, icon, qrcformat)))
  501. # ------------------------------------------------------------------------------------------------------------
  502. # Handle some basic command-line arguments shared between all carla variants
  503. def handleInitialCommandLineArguments(file):
  504. initName = os.path.basename(file) if (file is not None and os.path.dirname(file) in PATH) else sys.argv[0]
  505. libPrefix = None
  506. readPrefixNext = False
  507. for arg in sys.argv[1:]:
  508. if arg.startswith("--with-appname="):
  509. initName = os.path.basename(arg.replace("--with-appname=", ""))
  510. elif arg.startswith("--with-libprefix="):
  511. libPrefix = arg.replace("--with-libprefix=", "")
  512. elif arg.startswith("--osc-gui="):
  513. gCarla.nogui = int(arg.replace("--osc-gui=", ""))
  514. elif arg.startswith("--cnprefix="):
  515. gCarla.cnprefix = arg.replace("--cnprefix=", "")
  516. elif arg == "--cnprefix":
  517. readPrefixNext = True
  518. elif arg == "--gdb":
  519. pass
  520. elif arg in ("-n", "--n", "-no-gui", "--no-gui", "-nogui", "--nogui"):
  521. gCarla.nogui = True
  522. elif MACOS and arg.startswith("-psn_"):
  523. pass
  524. elif arg in ("-h", "--h", "-help", "--help"):
  525. print("Usage: %s [OPTION]... [FILE|URL]" % initName)
  526. print("")
  527. print(" where FILE can be a Carla project or preset file to be loaded, or URL if using Carla-Control")
  528. print("")
  529. print(" and OPTION can be one or more of the following:")
  530. print("")
  531. print(" --cnprefix\t Set a prefix for client names in multi-client mode.")
  532. if not isinstance(gCarla.nogui, int):
  533. print(" --gdb \t Run Carla inside gdb.")
  534. print(" -n,--no-gui \t Run Carla headless, don't show UI.")
  535. print("")
  536. print(" -h,--help \t Print this help text and exit.")
  537. print(" -v,--version \t Print version information and exit.")
  538. print("")
  539. if isinstance(gCarla.nogui, int):
  540. print("NOTE: when using %s the FILE is only valid the first time the backend is started" % initName)
  541. sys.exit(1)
  542. sys.exit(0)
  543. elif arg in ("-v", "--v", "-version", "--version"):
  544. pathBinaries, pathResources = getPaths(libPrefix)
  545. print("Using Carla version %s" % VERSION)
  546. print(" Python version: %s" % sys.version.split(" ",1)[0])
  547. print(" Qt version: %s" % QT_VERSION_STR)
  548. print(" PyQt version: %s" % PYQT_VERSION_STR)
  549. print(" Binary dir: %s" % pathBinaries)
  550. print(" Resources dir: %s" % pathResources)
  551. sys.exit(1 if gCarla.nogui else 0)
  552. elif readPrefixNext:
  553. readPrefixNext = False
  554. gCarla.cnprefix = arg
  555. if gCarla.nogui and isinstance(gCarla.nogui, int):
  556. if os.fork():
  557. # pylint: disable=protected-access
  558. os._exit(0)
  559. # pylint: enable=protected-access
  560. else:
  561. os.setsid()
  562. return (initName, libPrefix)
  563. # ------------------------------------------------------------------------------------------------------------
  564. # Get initial project file (as passed in the command-line parameters)
  565. def getInitialProjectFile(skipExistCheck = False):
  566. # NOTE: PyQt mishandles unicode characters, we directly use sys.argv instead of qApp->arguments()
  567. # see https://riverbankcomputing.com/pipermail/pyqt/2015-January/035395.html
  568. args = sys.argv[1:]
  569. readPrefixNext = False
  570. for arg in args:
  571. if readPrefixNext:
  572. readPrefixNext = False
  573. continue
  574. if arg.startswith("--cnprefix="):
  575. continue
  576. if arg.startswith("--osc-gui="):
  577. continue
  578. if arg.startswith("--with-appname="):
  579. continue
  580. if arg.startswith("--with-libprefix="):
  581. continue
  582. if arg == "--cnprefix":
  583. readPrefixNext = True
  584. continue
  585. if arg in ("-n", "--n", "-no-gui", "--no-gui", "-nogui", "--nogui", "--gdb"):
  586. continue
  587. if MACOS and arg.startswith("-psn_"):
  588. continue
  589. arg = os.path.expanduser(arg)
  590. if skipExistCheck or os.path.exists(arg):
  591. return arg
  592. return None
  593. # ------------------------------------------------------------------------------------------------------------
  594. # Get paths (binaries, resources)
  595. def getPaths(libPrefix = None):
  596. CWDl = CWD.lower()
  597. # adjust for special distros
  598. libdir = os.path.basename(os.path.normpath(X_LIBDIR_X)) if X_LIBDIR_X else "lib"
  599. datadir = os.path.basename(os.path.normpath(X_DATADIR_X)) if X_DATADIR_X else "share"
  600. # standalone, installed system-wide linux
  601. if libPrefix is not None:
  602. pathBinaries = os.path.join(libPrefix, libdir, "carla")
  603. pathResources = os.path.join(libPrefix, datadir, "carla", "resources")
  604. # standalone, local source
  605. elif CWDl.endswith("frontend"):
  606. pathBinaries = os.path.abspath(os.path.join(CWD, "..", "..", "bin"))
  607. pathResources = os.path.join(pathBinaries, "resources")
  608. # plugin
  609. elif CWDl.endswith("resources"):
  610. # installed system-wide linux
  611. if CWDl.endswith("/share/carla/resources"):
  612. pathBinaries = os.path.abspath(os.path.join(CWD, "..", "..", "..", libdir, "carla"))
  613. pathResources = CWD
  614. # local source
  615. elif CWDl.endswith("native-plugins%sresources" % os.sep):
  616. pathBinaries = os.path.abspath(os.path.join(CWD, "..", "..", "..", "bin"))
  617. pathResources = CWD
  618. # other
  619. else:
  620. pathBinaries = os.path.abspath(os.path.join(CWD, ".."))
  621. pathResources = CWD
  622. # everything else
  623. else:
  624. pathBinaries = CWD
  625. pathResources = os.path.join(pathBinaries, "resources")
  626. return (pathBinaries, pathResources)
  627. # ------------------------------------------------------------------------------------------------------------
  628. # Signal handler
  629. # TODO move to carla_host.py or something
  630. def signalHandler(sig, frame):
  631. if sig in (SIGINT, SIGTERM):
  632. gCarla.term = True
  633. if gCarla.gui is not None:
  634. gCarla.gui.SIGTERM.emit()
  635. elif haveSIGUSR1 and sig == SIGUSR1:
  636. if gCarla.gui is not None:
  637. gCarla.gui.SIGUSR1.emit()
  638. def setUpSignals():
  639. signal(SIGINT, signalHandler)
  640. signal(SIGTERM, signalHandler)
  641. if not haveSIGUSR1:
  642. return
  643. signal(SIGUSR1, signalHandler)
  644. # ------------------------------------------------------------------------------------------------------------
  645. # QLineEdit and QPushButton combo
  646. def getAndSetPath(parent, lineEdit):
  647. newPath = QFileDialog.getExistingDirectory(parent, parent.tr("Set Path"), lineEdit.text(), QFileDialog.ShowDirsOnly)
  648. if newPath:
  649. lineEdit.setText(newPath)
  650. return newPath
  651. # ------------------------------------------------------------------------------------------------------------
  652. # Backwards-compatible horizontalAdvance/width call, depending on Qt version
  653. def fontMetricsHorizontalAdvance(fontMetrics, string):
  654. if QT_VERSION >= 0x51100:
  655. return fontMetrics.horizontalAdvance(string)
  656. return fontMetrics.width(string)
  657. # ------------------------------------------------------------------------------------------------------------
  658. # Custom QMessageBox which resizes itself to fit text
  659. class QMessageBoxWithBetterWidth(QMessageBox):
  660. def __init__(self, parent):
  661. QMessageBox.__init__(self, parent)
  662. def showEvent(self, event):
  663. fontMetrics = self.fontMetrics()
  664. lines = self.text().strip().split("\n") + self.informativeText().strip().split("\n")
  665. if lines:
  666. width = 0
  667. for line in lines:
  668. width = max(fontMetricsHorizontalAdvance(fontMetrics, line), width)
  669. self.layout().setColumnMinimumWidth(2, width + 12)
  670. QMessageBox.showEvent(self, event)
  671. # ------------------------------------------------------------------------------------------------------------
  672. # Safer QSettings class, which does not throw if type mismatches
  673. class QSafeSettings(QSettings):
  674. def value(self, key, defaultValue, valueType):
  675. if not isinstance(defaultValue, valueType):
  676. print("QSafeSettings.value() - defaultValue type mismatch for key", key)
  677. try:
  678. return QSettings.value(self, key, defaultValue, valueType)
  679. except:
  680. return defaultValue
  681. # ------------------------------------------------------------------------------------------------------------
  682. # Custom MessageBox
  683. # pylint: disable=too-many-arguments
  684. def CustomMessageBox(parent, icon, title, text,
  685. extraText="",
  686. buttons=QMessageBox.Yes|QMessageBox.No,
  687. defButton=QMessageBox.No):
  688. msgBox = QMessageBoxWithBetterWidth(parent)
  689. msgBox.setIcon(icon)
  690. msgBox.setWindowTitle(title)
  691. msgBox.setText(text)
  692. msgBox.setInformativeText(extraText)
  693. msgBox.setStandardButtons(buttons)
  694. msgBox.setDefaultButton(defButton)
  695. return msgBox.exec_()
  696. # pylint: enable=too-many-arguments
  697. # ------------------------------------------------------------------------------------------------------------