binding.cs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449
  1. //
  2. // TODO:
  3. // * FindNCurses needs to remove the old probing code
  4. // * Removal of that proxy code
  5. // * Need to implement reading pointers with the new API
  6. // * Can remove the manual Dlopen features
  7. // * initscr() diagnostics based on DLL can be fixed
  8. //
  9. // binding.cs.in: Core binding for curses.
  10. //
  11. // This file attempts to call into ncurses without relying on Mono's
  12. // dllmap, so it will work with .NET Core. This means that it needs
  13. // two sets of bindings, one for "ncurses" which works on OSX, and one
  14. // that works against "libncursesw.so.5" which is what you find on
  15. // assorted Linux systems.
  16. //
  17. // Additionally, I do not want to rely on an external native library
  18. // which is why all this pain to bind two separate ncurses is here.
  19. //
  20. // Authors:
  21. // Miguel de Icaza ([email protected])
  22. //
  23. // Copyright (C) 2007 Novell (http://www.novell.com)
  24. //
  25. // Permission is hereby granted, free of charge, to any person obtaining
  26. // a copy of this software and associated documentation files (the
  27. // "Software"), to deal in the Software without restriction, including
  28. // without limitation the rights to use, copy, modify, merge, publish,
  29. // distribute, sublicense, and/or sell copies of the Software, and to
  30. // permit persons to whom the Software is furnished to do so, subject to
  31. // the following conditions:
  32. //
  33. // The above copyright notice and this permission notice shall be
  34. // included in all copies or substantial portions of the Software.
  35. //
  36. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  37. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  38. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  39. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  40. // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  41. // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  42. // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  43. //
  44. using System;
  45. using System.IO;
  46. using System.Runtime.InteropServices;
  47. using Mono.Terminal.Internal;
  48. namespace Unix.Terminal {
  49. public partial class Curses {
  50. [StructLayout (LayoutKind.Sequential)]
  51. public struct MouseEvent {
  52. public short ID;
  53. public int X, Y, Z;
  54. public Event ButtonState;
  55. }
  56. static int lines, cols;
  57. static Window main_window;
  58. static IntPtr curses_handle, curscr_ptr, lines_ptr, cols_ptr;
  59. // If true, uses the DllImport into "ncurses", otherwise "libncursesw.so.5"
  60. static bool use_naked_driver;
  61. static UnmanagedLibrary curses_library;
  62. static NativeMethods methods;
  63. static void LoadMethods ()
  64. {
  65. var libs = UnmanagedLibrary.IsMacOSPlatform ? new string [] { "libncurses.dylib" } : new string [] { "libncursesw.so.6", "libncursesw.so.5" };
  66. curses_library = new UnmanagedLibrary (libs, false);
  67. methods = new NativeMethods (curses_library);
  68. }
  69. static void FindNCurses ()
  70. {
  71. LoadMethods ();
  72. curses_handle = methods.UnmanagedLibrary.NativeLibraryHandle;
  73. stdscr = read_static_ptr ("stdscr");
  74. curscr_ptr = get_ptr ("curscr");
  75. lines_ptr = get_ptr ("LINES");
  76. cols_ptr = get_ptr ("COLS");
  77. }
  78. static public Window initscr ()
  79. {
  80. FindNCurses ();
  81. main_window = new Window (methods.initscr ());
  82. try {
  83. console_sharp_get_dims (out lines, out cols);
  84. } catch (DllNotFoundException){
  85. endwin ();
  86. Console.Error.WriteLine ("Unable to find the @MONO_CURSES@ native library\n" +
  87. "this is different than the managed mono-curses.dll\n\n" +
  88. "Typically you need to install to a LD_LIBRARY_PATH directory\n" +
  89. "or DYLD_LIBRARY_PATH directory or run /sbin/ldconfig");
  90. Environment.Exit (1);
  91. }
  92. return main_window;
  93. }
  94. public static int Lines {
  95. get {
  96. return lines;
  97. }
  98. }
  99. public static int Cols {
  100. get {
  101. return cols;
  102. }
  103. }
  104. //
  105. // Returns true if the window changed since the last invocation, as a
  106. // side effect, the Lines and Cols properties are updated
  107. //
  108. public static bool CheckWinChange ()
  109. {
  110. int l, c;
  111. console_sharp_get_dims (out l, out c);
  112. if (l != lines || c != cols){
  113. lines = l;
  114. cols = c;
  115. return true;
  116. }
  117. return false;
  118. }
  119. public static int addstr (string format, params object [] args)
  120. {
  121. var s = string.Format (format, args);
  122. return addstr (s);
  123. }
  124. static char [] r = new char [1];
  125. //
  126. // Have to wrap the native addch, as it can not
  127. // display unicode characters, we have to use addstr
  128. // for that. but we need addch to render special ACS
  129. // characters
  130. //
  131. public static int addch (int ch)
  132. {
  133. if (ch < 127 || ch > 0xffff )
  134. return methods.addch (ch);
  135. char c = (char) ch;
  136. return addstr (new String (c, 1));
  137. }
  138. static IntPtr stdscr;
  139. static IntPtr get_ptr (string key)
  140. {
  141. var ptr = curses_library.LoadSymbol (key);
  142. if (ptr == IntPtr.Zero)
  143. throw new Exception ("Could not load the key " + key);
  144. return ptr;
  145. }
  146. internal static IntPtr read_static_ptr (string key)
  147. {
  148. var ptr = get_ptr (key);
  149. return Marshal.ReadIntPtr (ptr);
  150. }
  151. internal static IntPtr console_sharp_get_stdscr () => stdscr;
  152. internal static IntPtr console_sharp_get_curscr ()
  153. {
  154. return Marshal.ReadIntPtr (curscr_ptr);
  155. }
  156. internal static void console_sharp_get_dims (out int lines, out int cols)
  157. {
  158. lines = Marshal.ReadInt32 (lines_ptr);
  159. cols = Marshal.ReadInt32 (cols_ptr);
  160. }
  161. public static Event mousemask (Event newmask, out Event oldmask)
  162. {
  163. IntPtr e;
  164. var ret = (Event) (methods.mousemask ((IntPtr) newmask, out e));
  165. oldmask = (Event) e;
  166. return ret;
  167. }
  168. // We encode ESC + char (what Alt-char generates) as 0x2000 + char
  169. public const int KeyAlt = 0x2000;
  170. static public int IsAlt (int key)
  171. {
  172. if ((key & KeyAlt) != 0)
  173. return key & ~KeyAlt;
  174. return 0;
  175. }
  176. public static int StartColor () => methods.start_color ();
  177. public static bool HasColors => methods.has_colors ();
  178. public static int InitColorPair (short pair, short foreground, short background) => methods.init_pair (pair, foreground, background);
  179. public static int UseDefaultColors () => methods.use_default_colors ();
  180. public static int ColorPairs => methods.COLOR_PAIRS();
  181. //
  182. // The proxy methods to call into each version
  183. //
  184. static public int endwin () => methods.endwin ();
  185. static public bool isendwin () => methods.isendwin ();
  186. static public int cbreak () => methods.cbreak ();
  187. static public int nocbreak () => methods.nocbreak ();
  188. static public int echo () => methods.echo ();
  189. static public int noecho () => methods.noecho ();
  190. static public int halfdelay (int t) => methods.halfdelay (t);
  191. static public int raw () => methods.raw ();
  192. static public int noraw () => methods.noraw ();
  193. static public void noqiflush () => methods.noqiflush ();
  194. static public void qiflush () => methods.qiflush ();
  195. static public int typeahead (IntPtr fd) => methods.typeahead (fd);
  196. static public int timeout (int delay) => methods.timeout (delay);
  197. static public int wtimeout (IntPtr win, int delay) => methods.wtimeout (win, delay);
  198. static public int notimeout (IntPtr win, bool bf) => methods.notimeout (win, bf);
  199. static public int keypad (IntPtr win, bool bf) => methods.keypad (win, bf);
  200. static public int meta (IntPtr win, bool bf) => methods.meta (win, bf);
  201. static public int intrflush (IntPtr win, bool bf) => methods.intrflush (win, bf);
  202. static public int clearok (IntPtr win, bool bf) => methods.clearok (win, bf);
  203. static public int idlok (IntPtr win, bool bf) => methods.idlok (win, bf);
  204. static public void idcok (IntPtr win, bool bf) => methods.idcok (win, bf);
  205. static public void immedok (IntPtr win, bool bf) => methods.immedok (win, bf);
  206. static public int leaveok (IntPtr win, bool bf) => methods.leaveok (win, bf);
  207. static public int wsetscrreg (IntPtr win, int top, int bot) => methods.wsetscrreg (win, top, bot);
  208. static public int scrollok (IntPtr win, bool bf) => methods.scrollok (win, bf);
  209. static public int nl() => methods.nl();
  210. static public int nonl() => methods.nonl();
  211. static public int setscrreg (int top, int bot) => methods.setscrreg (top, bot);
  212. static public int refresh () => methods.refresh ();
  213. static public int doupdate() => methods.doupdate();
  214. static public int wrefresh (IntPtr win) => methods.wrefresh (win);
  215. static public int redrawwin (IntPtr win) => methods.redrawwin (win);
  216. //static public int wredrawwin (IntPtr win, int beg_line, int num_lines) => methods.wredrawwin (win, beg_line, num_lines);
  217. static public int wnoutrefresh (IntPtr win) => methods.wnoutrefresh (win);
  218. static public int move (int line, int col) => methods.move (line, col);
  219. //static public int addch (int ch) => methods.addch (ch);
  220. static public int addstr (string s) => methods.addstr (s);
  221. static public int wmove (IntPtr win, int line, int col) => methods.wmove (win, line, col);
  222. static public int waddch (IntPtr win, int ch) => methods.waddch (win, ch);
  223. static public int attron (int attrs) => methods.attron (attrs);
  224. static public int attroff (int attrs) => methods.attroff (attrs);
  225. static public int attrset (int attrs) => methods.attrset (attrs);
  226. static public int getch () => methods.getch ();
  227. static public int get_wch (out int sequence) => methods.get_wch (out sequence);
  228. static public int ungetch (int ch) => methods.ungetch (ch);
  229. static public int mvgetch (int y, int x) => methods.mvgetch (y, x);
  230. static public bool has_colors () => methods.has_colors ();
  231. static public int start_color () => methods.start_color ();
  232. static public int init_pair (short pair, short f, short b) => methods.init_pair (pair, f, b);
  233. static public int use_default_colors () => methods.use_default_colors ();
  234. static public int COLOR_PAIRS() => methods.COLOR_PAIRS();
  235. static public uint getmouse (out MouseEvent ev) => methods.getmouse (out ev);
  236. static public uint ungetmouse (ref MouseEvent ev) => methods.ungetmouse (ref ev);
  237. static public int mouseinterval (int interval) => methods.mouseinterval (interval);
  238. }
  239. internal class Delegates {
  240. public delegate IntPtr initscr ();
  241. public delegate int endwin ();
  242. public delegate bool isendwin ();
  243. public delegate int cbreak ();
  244. public delegate int nocbreak ();
  245. public delegate int echo ();
  246. public delegate int noecho ();
  247. public delegate int halfdelay (int t);
  248. public delegate int raw ();
  249. public delegate int noraw ();
  250. public delegate void noqiflush ();
  251. public delegate void qiflush ();
  252. public delegate int typeahead (IntPtr fd);
  253. public delegate int timeout (int delay);
  254. public delegate int wtimeout (IntPtr win, int delay);
  255. public delegate int notimeout (IntPtr win, bool bf);
  256. public delegate int keypad (IntPtr win, bool bf);
  257. public delegate int meta (IntPtr win, bool bf);
  258. public delegate int intrflush (IntPtr win, bool bf);
  259. public delegate int clearok (IntPtr win, bool bf);
  260. public delegate int idlok (IntPtr win, bool bf);
  261. public delegate void idcok (IntPtr win, bool bf);
  262. public delegate void immedok (IntPtr win, bool bf);
  263. public delegate int leaveok (IntPtr win, bool bf);
  264. public delegate int wsetscrreg (IntPtr win, int top, int bot);
  265. public delegate int scrollok (IntPtr win, bool bf);
  266. public delegate int nl ();
  267. public delegate int nonl ();
  268. public delegate int setscrreg (int top, int bot);
  269. public delegate int refresh ();
  270. public delegate int doupdate ();
  271. public delegate int wrefresh (IntPtr win);
  272. public delegate int redrawwin (IntPtr win);
  273. //public delegate int wredrawwin (IntPtr win, int beg_line, int num_lines);
  274. public delegate int wnoutrefresh (IntPtr win);
  275. public delegate int move (int line, int col);
  276. public delegate int addch (int ch);
  277. public delegate int addstr (string s);
  278. public delegate int wmove (IntPtr win, int line, int col);
  279. public delegate int waddch (IntPtr win, int ch);
  280. public delegate int attron (int attrs);
  281. public delegate int attroff (int attrs);
  282. public delegate int attrset (int attrs);
  283. public delegate int getch ();
  284. public delegate int get_wch (out int sequence);
  285. public delegate int ungetch (int ch);
  286. public delegate int mvgetch (int y, int x);
  287. public delegate bool has_colors ();
  288. public delegate int start_color ();
  289. public delegate int init_pair (short pair, short f, short b);
  290. public delegate int use_default_colors ();
  291. public delegate int COLOR_PAIRS ();
  292. public delegate uint getmouse (out Curses.MouseEvent ev);
  293. public delegate uint ungetmouse (ref Curses.MouseEvent ev);
  294. public delegate int mouseinterval (int interval);
  295. public delegate IntPtr mousemask (IntPtr newmask, out IntPtr oldMask);
  296. }
  297. internal class NativeMethods {
  298. public readonly Delegates.initscr initscr;
  299. public readonly Delegates.endwin endwin;
  300. public readonly Delegates.isendwin isendwin;
  301. public readonly Delegates.cbreak cbreak;
  302. public readonly Delegates.nocbreak nocbreak;
  303. public readonly Delegates.echo echo;
  304. public readonly Delegates.noecho noecho;
  305. public readonly Delegates.halfdelay halfdelay;
  306. public readonly Delegates.raw raw;
  307. public readonly Delegates.noraw noraw;
  308. public readonly Delegates.noqiflush noqiflush;
  309. public readonly Delegates.qiflush qiflush;
  310. public readonly Delegates.typeahead typeahead;
  311. public readonly Delegates.timeout timeout;
  312. public readonly Delegates.wtimeout wtimeout;
  313. public readonly Delegates.notimeout notimeout;
  314. public readonly Delegates.keypad keypad;
  315. public readonly Delegates.meta meta;
  316. public readonly Delegates.intrflush intrflush;
  317. public readonly Delegates.clearok clearok;
  318. public readonly Delegates.idlok idlok;
  319. public readonly Delegates.idcok idcok;
  320. public readonly Delegates.immedok immedok;
  321. public readonly Delegates.leaveok leaveok;
  322. public readonly Delegates.wsetscrreg wsetscrreg;
  323. public readonly Delegates.scrollok scrollok;
  324. public readonly Delegates.nl nl;
  325. public readonly Delegates.nonl nonl;
  326. public readonly Delegates.setscrreg setscrreg;
  327. public readonly Delegates.refresh refresh;
  328. public readonly Delegates.doupdate doupdate;
  329. public readonly Delegates.wrefresh wrefresh;
  330. public readonly Delegates.redrawwin redrawwin;
  331. //public readonly Delegates.wredrawwin wredrawwin;
  332. public readonly Delegates.wnoutrefresh wnoutrefresh;
  333. public readonly Delegates.move move;
  334. public readonly Delegates.addch addch;
  335. public readonly Delegates.addstr addstr;
  336. public readonly Delegates.wmove wmove;
  337. public readonly Delegates.waddch waddch;
  338. public readonly Delegates.attron attron;
  339. public readonly Delegates.attroff attroff;
  340. public readonly Delegates.attrset attrset;
  341. public readonly Delegates.getch getch;
  342. public readonly Delegates.get_wch get_wch;
  343. public readonly Delegates.ungetch ungetch;
  344. public readonly Delegates.mvgetch mvgetch;
  345. public readonly Delegates.has_colors has_colors;
  346. public readonly Delegates.start_color start_color;
  347. public readonly Delegates.init_pair init_pair;
  348. public readonly Delegates.use_default_colors use_default_colors;
  349. public readonly Delegates.COLOR_PAIRS COLOR_PAIRS;
  350. public readonly Delegates.getmouse getmouse;
  351. public readonly Delegates.ungetmouse ungetmouse;
  352. public readonly Delegates.mouseinterval mouseinterval;
  353. public readonly Delegates.mousemask mousemask;
  354. public UnmanagedLibrary UnmanagedLibrary;
  355. public NativeMethods (UnmanagedLibrary lib)
  356. {
  357. this.UnmanagedLibrary = lib;
  358. initscr = lib.GetNativeMethodDelegate<Delegates.initscr> ("initscr");
  359. endwin = lib.GetNativeMethodDelegate<Delegates.endwin> ("endwin");
  360. isendwin = lib.GetNativeMethodDelegate<Delegates.isendwin> ("isendwin");
  361. cbreak = lib.GetNativeMethodDelegate<Delegates.cbreak> ("cbreak");
  362. nocbreak = lib.GetNativeMethodDelegate<Delegates.nocbreak> ("nocbreak");
  363. echo = lib.GetNativeMethodDelegate<Delegates.echo> ("echo");
  364. noecho = lib.GetNativeMethodDelegate<Delegates.noecho> ("noecho");
  365. halfdelay = lib.GetNativeMethodDelegate<Delegates.halfdelay> ("halfdelay");
  366. raw = lib.GetNativeMethodDelegate<Delegates.raw> ("raw");
  367. noraw = lib.GetNativeMethodDelegate<Delegates.noraw> ("noraw");
  368. noqiflush = lib.GetNativeMethodDelegate<Delegates.noqiflush> ("noqiflush");
  369. qiflush = lib.GetNativeMethodDelegate<Delegates.qiflush> ("qiflush");
  370. typeahead = lib.GetNativeMethodDelegate<Delegates.typeahead> ("typeahead");
  371. timeout = lib.GetNativeMethodDelegate<Delegates.timeout> ("timeout");
  372. wtimeout = lib.GetNativeMethodDelegate<Delegates.wtimeout> ("wtimeout");
  373. notimeout = lib.GetNativeMethodDelegate<Delegates.notimeout> ("notimeout");
  374. keypad = lib.GetNativeMethodDelegate<Delegates.keypad> ("keypad");
  375. meta = lib.GetNativeMethodDelegate<Delegates.meta> ("meta");
  376. intrflush = lib.GetNativeMethodDelegate<Delegates.intrflush> ("intrflush");
  377. clearok = lib.GetNativeMethodDelegate<Delegates.clearok> ("clearok");
  378. idlok = lib.GetNativeMethodDelegate<Delegates.idlok> ("idlok");
  379. idcok = lib.GetNativeMethodDelegate<Delegates.idcok> ("idcok");
  380. immedok = lib.GetNativeMethodDelegate<Delegates.immedok> ("immedok");
  381. leaveok = lib.GetNativeMethodDelegate<Delegates.leaveok> ("leaveok");
  382. wsetscrreg = lib.GetNativeMethodDelegate<Delegates.wsetscrreg> ("wsetscrreg");
  383. scrollok = lib.GetNativeMethodDelegate<Delegates.scrollok> ("scrollok");
  384. nl = lib.GetNativeMethodDelegate<Delegates.nl> ("nl");
  385. nonl = lib.GetNativeMethodDelegate<Delegates.nonl> ("nonl");
  386. setscrreg = lib.GetNativeMethodDelegate<Delegates.setscrreg> ("setscrreg");
  387. refresh = lib.GetNativeMethodDelegate<Delegates.refresh> ("refresh");
  388. doupdate = lib.GetNativeMethodDelegate<Delegates.doupdate> ("doupdate");
  389. wrefresh = lib.GetNativeMethodDelegate<Delegates.wrefresh> ("wrefresh");
  390. redrawwin = lib.GetNativeMethodDelegate<Delegates.redrawwin> ("redrawwin");
  391. //wredrawwin = lib.GetNativeMethodDelegate<Delegates.wredrawwin> ("wredrawwin");
  392. wnoutrefresh = lib.GetNativeMethodDelegate<Delegates.wnoutrefresh> ("wnoutrefresh");
  393. move = lib.GetNativeMethodDelegate<Delegates.move> ("move");
  394. addch = lib.GetNativeMethodDelegate<Delegates.addch> ("addch");
  395. addstr = lib.GetNativeMethodDelegate<Delegates.addstr> ("addstr");
  396. wmove = lib.GetNativeMethodDelegate<Delegates.wmove> ("wmove");
  397. waddch = lib.GetNativeMethodDelegate<Delegates.waddch> ("waddch");
  398. attron = lib.GetNativeMethodDelegate<Delegates.attron> ("attron");
  399. attroff = lib.GetNativeMethodDelegate<Delegates.attroff> ("attroff");
  400. attrset = lib.GetNativeMethodDelegate<Delegates.attrset> ("attrset");
  401. getch = lib.GetNativeMethodDelegate<Delegates.getch> ("getch");
  402. get_wch = lib.GetNativeMethodDelegate<Delegates.get_wch> ("get_wch");
  403. ungetch = lib.GetNativeMethodDelegate<Delegates.ungetch> ("ungetch");
  404. mvgetch = lib.GetNativeMethodDelegate<Delegates.mvgetch> ("mvgetch");
  405. has_colors = lib.GetNativeMethodDelegate<Delegates.has_colors> ("has_colors");
  406. start_color = lib.GetNativeMethodDelegate<Delegates.start_color> ("start_color");
  407. init_pair = lib.GetNativeMethodDelegate<Delegates.init_pair> ("init_pair");
  408. use_default_colors = lib.GetNativeMethodDelegate<Delegates.use_default_colors> ("use_default_colors");
  409. COLOR_PAIRS = lib.GetNativeMethodDelegate<Delegates.COLOR_PAIRS> ("COLOR_PAIRS");
  410. getmouse = lib.GetNativeMethodDelegate<Delegates.getmouse> ("getmouse");
  411. ungetmouse = lib.GetNativeMethodDelegate<Delegates.ungetmouse> ("ungetmouse");
  412. mouseinterval = lib.GetNativeMethodDelegate<Delegates.mouseinterval> ("mouseinterval");
  413. mousemask = lib.GetNativeMethodDelegate<Delegates.mousemask> ("mousemask");
  414. }
  415. }
  416. }