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.

337 lines
11KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Carla Backend utils
  4. # Copyright (C) 2011-2018 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. from os import environ
  20. from sys import argv
  21. # ------------------------------------------------------------------------------------------------------------
  22. # Imports (Custom)
  23. from carla_backend import *
  24. # ------------------------------------------------------------------------------------------------------------
  25. def getPluginTypeAsString(ptype):
  26. if ptype == PLUGIN_NONE:
  27. return "NONE"
  28. if ptype == PLUGIN_INTERNAL:
  29. return "Internal"
  30. if ptype == PLUGIN_LADSPA:
  31. return "LADSPA"
  32. if ptype == PLUGIN_DSSI:
  33. return "DSSI"
  34. if ptype == PLUGIN_LV2:
  35. return "LV2"
  36. if ptype == PLUGIN_VST2:
  37. return "VST2"
  38. if ptype == PLUGIN_SF2:
  39. return "SF2"
  40. if ptype == PLUGIN_SFZ:
  41. return "SFZ"
  42. if ptype == PLUGIN_JACK:
  43. return "JACK"
  44. print("getPluginTypeAsString(%i) - invalid type" % ptype);
  45. return "Unknown"
  46. def getPluginTypeFromString(stype):
  47. if not stype:
  48. return PLUGIN_NONE
  49. stype = stype.lower()
  50. if stype == "none":
  51. return PLUGIN_NONE
  52. if stype in ("internal", "native"):
  53. return PLUGIN_INTERNAL
  54. if stype == "ladspa":
  55. return PLUGIN_LADSPA
  56. if stype == "dssi":
  57. return PLUGIN_DSSI
  58. if stype == "lv2":
  59. return PLUGIN_LV2
  60. if stype in ("vst2", "vst"):
  61. return PLUGIN_VST2
  62. if stype == "sf2":
  63. return PLUGIN_SF2
  64. if stype == "sfz":
  65. return PLUGIN_SFZ
  66. if stype == "jack":
  67. return PLUGIN_JACK
  68. print("getPluginTypeFromString(\"%s\") - invalid string type" % stype)
  69. return PLUGIN_NONE
  70. # ------------------------------------------------------------------------------------------------------------
  71. # Carla Utils API (C stuff)
  72. CarlaPipeClientHandle = c_void_p
  73. CarlaPipeCallbackFunc = CFUNCTYPE(None, c_void_p, c_char_p)
  74. # Information about an internal Carla plugin.
  75. # @see carla_get_cached_plugin_info()
  76. class CarlaCachedPluginInfo(Structure):
  77. _fields_ = [
  78. # Wherever the data in this struct is valid.
  79. # For performance reasons, plugins are only checked on request,
  80. # and as such, the count vs number of really valid plugins might not match.
  81. # Use this field to skip on plugins which cannot be loaded in Carla.
  82. ("valid", c_bool),
  83. # Plugin category.
  84. ("category", c_enum),
  85. # Plugin hints.
  86. # @see PluginHints
  87. ("hints", c_uint),
  88. # Number of audio inputs.
  89. ("audioIns", c_uint32),
  90. # Number of audio outputs.
  91. ("audioOuts", c_uint32),
  92. # Number of MIDI inputs.
  93. ("midiIns", c_uint32),
  94. # Number of MIDI outputs.
  95. ("midiOuts", c_uint32),
  96. # Number of input parameters.
  97. ("parameterIns", c_uint32),
  98. # Number of output parameters.
  99. ("parameterOuts", c_uint32),
  100. # Plugin name.
  101. ("name", c_char_p),
  102. # Plugin label.
  103. ("label", c_char_p),
  104. # Plugin author/maker.
  105. ("maker", c_char_p),
  106. # Plugin copyright/license.
  107. ("copyright", c_char_p)
  108. ]
  109. # ------------------------------------------------------------------------------------------------------------
  110. # Carla Utils API (Python compatible stuff)
  111. # @see CarlaCachedPluginInfo
  112. PyCarlaCachedPluginInfo = {
  113. 'valid': False,
  114. 'category': PLUGIN_CATEGORY_NONE,
  115. 'hints': 0x0,
  116. 'audioIns': 0,
  117. 'audioOuts': 0,
  118. 'midiIns': 0,
  119. 'midiOuts': 0,
  120. 'parameterIns': 0,
  121. 'parameterOuts': 0,
  122. 'name': "",
  123. 'label': "",
  124. 'maker': "",
  125. 'copyright': ""
  126. }
  127. # ------------------------------------------------------------------------------------------------------------
  128. # Carla Utils object using a DLL
  129. class CarlaUtils(object):
  130. def __init__(self, filename):
  131. object.__init__(self)
  132. self.lib = cdll.LoadLibrary(filename)
  133. #self.lib = CDLL(filename, RTLD_GLOBAL)
  134. self.lib.carla_get_complete_license_text.argtypes = None
  135. self.lib.carla_get_complete_license_text.restype = c_char_p
  136. self.lib.carla_get_supported_file_extensions.argtypes = None
  137. self.lib.carla_get_supported_file_extensions.restype = POINTER(c_char_p)
  138. self.lib.carla_get_supported_features.argtypes = None
  139. self.lib.carla_get_supported_features.restype = POINTER(c_char_p)
  140. self.lib.carla_get_cached_plugin_count.argtypes = [c_enum, c_char_p]
  141. self.lib.carla_get_cached_plugin_count.restype = c_uint
  142. self.lib.carla_get_cached_plugin_info.argtypes = [c_enum, c_uint]
  143. self.lib.carla_get_cached_plugin_info.restype = POINTER(CarlaCachedPluginInfo)
  144. self.lib.carla_fflush.argtypes = [c_bool]
  145. self.lib.carla_fflush.restype = None
  146. self.lib.carla_fputs.argtypes = [c_bool, c_char_p]
  147. self.lib.carla_fputs.restype = None
  148. self.lib.carla_set_process_name.argtypes = [c_char_p]
  149. self.lib.carla_set_process_name.restype = None
  150. self.lib.carla_pipe_client_new.argtypes = [POINTER(c_char_p), CarlaPipeCallbackFunc, c_void_p]
  151. self.lib.carla_pipe_client_new.restype = CarlaPipeClientHandle
  152. self.lib.carla_pipe_client_idle.argtypes = [CarlaPipeClientHandle]
  153. self.lib.carla_pipe_client_idle.restype = None
  154. self.lib.carla_pipe_client_is_running.argtypes = [CarlaPipeClientHandle]
  155. self.lib.carla_pipe_client_is_running.restype = c_bool
  156. self.lib.carla_pipe_client_lock.argtypes = [CarlaPipeClientHandle]
  157. self.lib.carla_pipe_client_lock.restype = None
  158. self.lib.carla_pipe_client_unlock.argtypes = [CarlaPipeClientHandle]
  159. self.lib.carla_pipe_client_unlock.restype = None
  160. self.lib.carla_pipe_client_readlineblock.argtypes = [CarlaPipeClientHandle, c_uint]
  161. self.lib.carla_pipe_client_readlineblock.restype = c_char_p
  162. self.lib.carla_pipe_client_write_msg.argtypes = [CarlaPipeClientHandle, c_char_p]
  163. self.lib.carla_pipe_client_write_msg.restype = c_bool
  164. self.lib.carla_pipe_client_write_and_fix_msg.argtypes = [CarlaPipeClientHandle, c_char_p]
  165. self.lib.carla_pipe_client_write_and_fix_msg.restype = c_bool
  166. self.lib.carla_pipe_client_flush.argtypes = [CarlaPipeClientHandle]
  167. self.lib.carla_pipe_client_flush.restype = c_bool
  168. self.lib.carla_pipe_client_flush_and_unlock.argtypes = [CarlaPipeClientHandle]
  169. self.lib.carla_pipe_client_flush_and_unlock.restype = c_bool
  170. self.lib.carla_pipe_client_destroy.argtypes = [CarlaPipeClientHandle]
  171. self.lib.carla_pipe_client_destroy.restype = None
  172. self.lib.carla_cocoa_get_window.argtypes = [c_uintptr]
  173. self.lib.carla_cocoa_get_window.restype = c_int
  174. # use _putenv on windows
  175. if not WINDOWS:
  176. self.msvcrt = None
  177. return
  178. self.msvcrt = cdll.msvcrt
  179. self.msvcrt._putenv.argtypes = [c_char_p]
  180. self.msvcrt._putenv.restype = None
  181. # --------------------------------------------------------------------------------------------------------
  182. # set environment variable
  183. def setenv(self, key, value):
  184. environ[key] = value
  185. if WINDOWS:
  186. keyvalue = "%s=%s" % (key, value)
  187. self.msvcrt._putenv(keyvalue.encode("utf-8"))
  188. # unset environment variable
  189. def unsetenv(self, key):
  190. if environ.get(key) is not None:
  191. environ.pop(key)
  192. if WINDOWS:
  193. keyrm = "%s=" % key
  194. self.msvcrt._putenv(keyrm.encode("utf-8"))
  195. # --------------------------------------------------------------------------------------------------------
  196. # Get the complete license text of used third-party code and features.
  197. # Returned string is in basic html format.
  198. def get_complete_license_text(self):
  199. return charPtrToString(self.lib.carla_get_complete_license_text())
  200. # Get the list of supported file extensions in carla_load_file().
  201. def get_supported_file_extensions(self):
  202. return charPtrPtrToStringList(self.lib.carla_get_supported_file_extensions())
  203. # Get the list of supported features in the current Carla build.
  204. def get_supported_features(self):
  205. return charPtrPtrToStringList(self.lib.carla_get_supported_features())
  206. # Get how many internal plugins are available.
  207. # Internal and LV2 plugin formats are cached and need to be discovered via this function.
  208. # Do not call this for any other plugin formats.
  209. def get_cached_plugin_count(self, ptype, pluginPath):
  210. return int(self.lib.carla_get_cached_plugin_count(ptype, pluginPath.encode("utf-8")))
  211. # Get information about a cached plugin.
  212. def get_cached_plugin_info(self, ptype, index):
  213. return structToDict(self.lib.carla_get_cached_plugin_info(ptype, index).contents)
  214. def fflush(self, err):
  215. self.lib.carla_fflush(err)
  216. def fputs(self, err, string):
  217. self.lib.carla_fputs(err, string.encode("utf-8"))
  218. def set_process_name(self, name):
  219. self.lib.carla_set_process_name(name.encode("utf-8"))
  220. def pipe_client_new(self, func):
  221. argc = len(argv)
  222. cagrvtype = c_char_p * argc
  223. cargv = cagrvtype()
  224. for i in range(argc):
  225. cargv[i] = c_char_p(argv[i].encode("utf-8"))
  226. self._pipeClientCallback = CarlaPipeCallbackFunc(func)
  227. return self.lib.carla_pipe_client_new(cargv, self._pipeClientCallback, None)
  228. def pipe_client_idle(self, handle):
  229. try:
  230. self.lib.carla_pipe_client_idle(handle)
  231. except OSError as e:
  232. print("pipe_client_idle", e)
  233. def pipe_client_is_running(self, handle):
  234. return bool(self.lib.carla_pipe_client_is_running(handle))
  235. def pipe_client_lock(self, handle):
  236. self.lib.carla_pipe_client_lock(handle)
  237. def pipe_client_unlock(self, handle):
  238. self.lib.carla_pipe_client_unlock(handle)
  239. def pipe_client_readlineblock(self, handle, timeout):
  240. return charPtrToString(self.lib.carla_pipe_client_readlineblock(handle, timeout))
  241. def pipe_client_write_msg(self, handle, msg):
  242. return bool(self.lib.carla_pipe_client_write_msg(handle, msg.encode("utf-8")))
  243. def pipe_client_write_and_fix_msg(self, handle, msg):
  244. return bool(self.lib.carla_pipe_client_write_and_fix_msg(handle, msg.encode("utf-8")))
  245. def pipe_client_flush(self, handle):
  246. return bool(self.lib.carla_pipe_client_flush(handle))
  247. def pipe_client_flush_and_unlock(self, handle):
  248. return bool(self.lib.carla_pipe_client_flush_and_unlock(handle))
  249. def pipe_client_destroy(self, handle):
  250. self.lib.carla_pipe_client_destroy(handle)
  251. def cocoa_get_window(self, winId):
  252. return self.lib.carla_cocoa_get_window(winId)
  253. # ------------------------------------------------------------------------------------------------------------