binding.cs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  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. [DllImport("libc")]
  64. public extern static int setlocale(int cate, [MarshalAs(UnmanagedType.LPStr)] string locale);
  65. static void LoadMethods ()
  66. {
  67. var libs = UnmanagedLibrary.IsMacOSPlatform ? new string [] { "libncurses.dylib" } : new string [] { "libncursesw.so.6", "libncursesw.so.5" };
  68. curses_library = new UnmanagedLibrary (libs, false);
  69. methods = new NativeMethods (curses_library);
  70. }
  71. static void FindNCurses ()
  72. {
  73. LoadMethods ();
  74. curses_handle = methods.UnmanagedLibrary.NativeLibraryHandle;
  75. stdscr = read_static_ptr ("stdscr");
  76. curscr_ptr = get_ptr ("curscr");
  77. lines_ptr = get_ptr ("LINES");
  78. cols_ptr = get_ptr ("COLS");
  79. }
  80. static public Window initscr ()
  81. {
  82. setlocale(LC_ALL, "");
  83. FindNCurses ();
  84. main_window = new Window (methods.initscr ());
  85. try {
  86. console_sharp_get_dims (out lines, out cols);
  87. } catch (DllNotFoundException){
  88. endwin ();
  89. Console.Error.WriteLine ("Unable to find the @MONO_CURSES@ native library\n" +
  90. "this is different than the managed mono-curses.dll\n\n" +
  91. "Typically you need to install to a LD_LIBRARY_PATH directory\n" +
  92. "or DYLD_LIBRARY_PATH directory or run /sbin/ldconfig");
  93. Environment.Exit (1);
  94. }
  95. return main_window;
  96. }
  97. public static int Lines {
  98. get {
  99. return lines;
  100. }
  101. }
  102. public static int Cols {
  103. get {
  104. return cols;
  105. }
  106. }
  107. //
  108. // Returns true if the window changed since the last invocation, as a
  109. // side effect, the Lines and Cols properties are updated
  110. //
  111. public static bool CheckWinChange ()
  112. {
  113. int l, c;
  114. console_sharp_get_dims (out l, out c);
  115. if (l != lines || c != cols){
  116. lines = l;
  117. cols = c;
  118. return true;
  119. }
  120. return false;
  121. }
  122. public static int addstr (string format, params object [] args)
  123. {
  124. var s = string.Format (format, args);
  125. return addwstr (s);
  126. }
  127. static char [] r = new char [1];
  128. //
  129. // Have to wrap the native addch, as it can not
  130. // display unicode characters, we have to use addstr
  131. // for that. but we need addch to render special ACS
  132. // characters
  133. //
  134. public static int addch (int ch)
  135. {
  136. if (ch < 127 || ch > 0xffff )
  137. return methods.addch (ch);
  138. char c = (char) ch;
  139. return addwstr (new String (c, 1));
  140. }
  141. static IntPtr stdscr;
  142. static IntPtr get_ptr (string key)
  143. {
  144. var ptr = curses_library.LoadSymbol (key);
  145. if (ptr == IntPtr.Zero)
  146. throw new Exception ("Could not load the key " + key);
  147. return ptr;
  148. }
  149. internal static IntPtr read_static_ptr (string key)
  150. {
  151. var ptr = get_ptr (key);
  152. return Marshal.ReadIntPtr (ptr);
  153. }
  154. internal static IntPtr console_sharp_get_stdscr () => stdscr;
  155. internal static IntPtr console_sharp_get_curscr ()
  156. {
  157. return Marshal.ReadIntPtr (curscr_ptr);
  158. }
  159. internal static void console_sharp_get_dims (out int lines, out int cols)
  160. {
  161. lines = Marshal.ReadInt32 (lines_ptr);
  162. cols = Marshal.ReadInt32 (cols_ptr);
  163. }
  164. public static Event mousemask (Event newmask, out Event oldmask)
  165. {
  166. IntPtr e;
  167. var ret = (Event) (methods.mousemask ((IntPtr) newmask, out e));
  168. oldmask = (Event) e;
  169. return ret;
  170. }
  171. // We encode ESC + char (what Alt-char generates) as 0x2000 + char
  172. public const int KeyAlt = 0x2000;
  173. static public int IsAlt (int key)
  174. {
  175. if ((key & KeyAlt) != 0)
  176. return key & ~KeyAlt;
  177. return 0;
  178. }
  179. public static int StartColor () => methods.start_color ();
  180. public static bool HasColors => methods.has_colors ();
  181. public static int InitColorPair (short pair, short foreground, short background) => methods.init_pair (pair, foreground, background);
  182. public static int UseDefaultColors () => methods.use_default_colors ();
  183. public static int ColorPairs => methods.COLOR_PAIRS();
  184. //
  185. // The proxy methods to call into each version
  186. //
  187. static public int endwin () => methods.endwin ();
  188. static public bool isendwin () => methods.isendwin ();
  189. static public int cbreak () => methods.cbreak ();
  190. static public int nocbreak () => methods.nocbreak ();
  191. static public int echo () => methods.echo ();
  192. static public int noecho () => methods.noecho ();
  193. static public int halfdelay (int t) => methods.halfdelay (t);
  194. static public int raw () => methods.raw ();
  195. static public int noraw () => methods.noraw ();
  196. static public void noqiflush () => methods.noqiflush ();
  197. static public void qiflush () => methods.qiflush ();
  198. static public int typeahead (IntPtr fd) => methods.typeahead (fd);
  199. static public int timeout (int delay) => methods.timeout (delay);
  200. static public int wtimeout (IntPtr win, int delay) => methods.wtimeout (win, delay);
  201. static public int notimeout (IntPtr win, bool bf) => methods.notimeout (win, bf);
  202. static public int keypad (IntPtr win, bool bf) => methods.keypad (win, bf);
  203. static public int meta (IntPtr win, bool bf) => methods.meta (win, bf);
  204. static public int intrflush (IntPtr win, bool bf) => methods.intrflush (win, bf);
  205. static public int clearok (IntPtr win, bool bf) => methods.clearok (win, bf);
  206. static public int idlok (IntPtr win, bool bf) => methods.idlok (win, bf);
  207. static public void idcok (IntPtr win, bool bf) => methods.idcok (win, bf);
  208. static public void immedok (IntPtr win, bool bf) => methods.immedok (win, bf);
  209. static public int leaveok (IntPtr win, bool bf) => methods.leaveok (win, bf);
  210. static public int wsetscrreg (IntPtr win, int top, int bot) => methods.wsetscrreg (win, top, bot);
  211. static public int scrollok (IntPtr win, bool bf) => methods.scrollok (win, bf);
  212. static public int nl() => methods.nl();
  213. static public int nonl() => methods.nonl();
  214. static public int setscrreg (int top, int bot) => methods.setscrreg (top, bot);
  215. static public int refresh () => methods.refresh ();
  216. static public int doupdate() => methods.doupdate();
  217. static public int wrefresh (IntPtr win) => methods.wrefresh (win);
  218. static public int redrawwin (IntPtr win) => methods.redrawwin (win);
  219. //static public int wredrawwin (IntPtr win, int beg_line, int num_lines) => methods.wredrawwin (win, beg_line, num_lines);
  220. static public int wnoutrefresh (IntPtr win) => methods.wnoutrefresh (win);
  221. static public int move (int line, int col) => methods.move (line, col);
  222. //static public int addch (int ch) => methods.addch (ch);
  223. static public int addwstr (string s) => methods.addwstr (s);
  224. static public int wmove (IntPtr win, int line, int col) => methods.wmove (win, line, col);
  225. static public int waddch (IntPtr win, int ch) => methods.waddch (win, ch);
  226. static public int attron (int attrs) => methods.attron (attrs);
  227. static public int attroff (int attrs) => methods.attroff (attrs);
  228. static public int attrset (int attrs) => methods.attrset (attrs);
  229. static public int getch () => methods.getch ();
  230. static public int get_wch (out int sequence) => methods.get_wch (out sequence);
  231. static public int ungetch (int ch) => methods.ungetch (ch);
  232. static public int mvgetch (int y, int x) => methods.mvgetch (y, x);
  233. static public bool has_colors () => methods.has_colors ();
  234. static public int start_color () => methods.start_color ();
  235. static public int init_pair (short pair, short f, short b) => methods.init_pair (pair, f, b);
  236. static public int use_default_colors () => methods.use_default_colors ();
  237. static public int COLOR_PAIRS() => methods.COLOR_PAIRS();
  238. static public uint getmouse (out MouseEvent ev) => methods.getmouse (out ev);
  239. static public uint ungetmouse (ref MouseEvent ev) => methods.ungetmouse (ref ev);
  240. static public int mouseinterval (int interval) => methods.mouseinterval (interval);
  241. }
  242. internal class Delegates {
  243. public delegate IntPtr initscr ();
  244. public delegate int endwin ();
  245. public delegate bool isendwin ();
  246. public delegate int cbreak ();
  247. public delegate int nocbreak ();
  248. public delegate int echo ();
  249. public delegate int noecho ();
  250. public delegate int halfdelay (int t);
  251. public delegate int raw ();
  252. public delegate int noraw ();
  253. public delegate void noqiflush ();
  254. public delegate void qiflush ();
  255. public delegate int typeahead (IntPtr fd);
  256. public delegate int timeout (int delay);
  257. public delegate int wtimeout (IntPtr win, int delay);
  258. public delegate int notimeout (IntPtr win, bool bf);
  259. public delegate int keypad (IntPtr win, bool bf);
  260. public delegate int meta (IntPtr win, bool bf);
  261. public delegate int intrflush (IntPtr win, bool bf);
  262. public delegate int clearok (IntPtr win, bool bf);
  263. public delegate int idlok (IntPtr win, bool bf);
  264. public delegate void idcok (IntPtr win, bool bf);
  265. public delegate void immedok (IntPtr win, bool bf);
  266. public delegate int leaveok (IntPtr win, bool bf);
  267. public delegate int wsetscrreg (IntPtr win, int top, int bot);
  268. public delegate int scrollok (IntPtr win, bool bf);
  269. public delegate int nl ();
  270. public delegate int nonl ();
  271. public delegate int setscrreg (int top, int bot);
  272. public delegate int refresh ();
  273. public delegate int doupdate ();
  274. public delegate int wrefresh (IntPtr win);
  275. public delegate int redrawwin (IntPtr win);
  276. //public delegate int wredrawwin (IntPtr win, int beg_line, int num_lines);
  277. public delegate int wnoutrefresh (IntPtr win);
  278. public delegate int move (int line, int col);
  279. public delegate int addch (int ch);
  280. public delegate int addwstr([MarshalAs(UnmanagedType.LPWStr)]string s);
  281. public delegate int wmove (IntPtr win, int line, int col);
  282. public delegate int waddch (IntPtr win, int ch);
  283. public delegate int attron (int attrs);
  284. public delegate int attroff (int attrs);
  285. public delegate int attrset (int attrs);
  286. public delegate int getch ();
  287. public delegate int get_wch (out int sequence);
  288. public delegate int ungetch (int ch);
  289. public delegate int mvgetch (int y, int x);
  290. public delegate bool has_colors ();
  291. public delegate int start_color ();
  292. public delegate int init_pair (short pair, short f, short b);
  293. public delegate int use_default_colors ();
  294. public delegate int COLOR_PAIRS ();
  295. public delegate uint getmouse (out Curses.MouseEvent ev);
  296. public delegate uint ungetmouse (ref Curses.MouseEvent ev);
  297. public delegate int mouseinterval (int interval);
  298. public delegate IntPtr mousemask (IntPtr newmask, out IntPtr oldMask);
  299. }
  300. internal class NativeMethods {
  301. public readonly Delegates.initscr initscr;
  302. public readonly Delegates.endwin endwin;
  303. public readonly Delegates.isendwin isendwin;
  304. public readonly Delegates.cbreak cbreak;
  305. public readonly Delegates.nocbreak nocbreak;
  306. public readonly Delegates.echo echo;
  307. public readonly Delegates.noecho noecho;
  308. public readonly Delegates.halfdelay halfdelay;
  309. public readonly Delegates.raw raw;
  310. public readonly Delegates.noraw noraw;
  311. public readonly Delegates.noqiflush noqiflush;
  312. public readonly Delegates.qiflush qiflush;
  313. public readonly Delegates.typeahead typeahead;
  314. public readonly Delegates.timeout timeout;
  315. public readonly Delegates.wtimeout wtimeout;
  316. public readonly Delegates.notimeout notimeout;
  317. public readonly Delegates.keypad keypad;
  318. public readonly Delegates.meta meta;
  319. public readonly Delegates.intrflush intrflush;
  320. public readonly Delegates.clearok clearok;
  321. public readonly Delegates.idlok idlok;
  322. public readonly Delegates.idcok idcok;
  323. public readonly Delegates.immedok immedok;
  324. public readonly Delegates.leaveok leaveok;
  325. public readonly Delegates.wsetscrreg wsetscrreg;
  326. public readonly Delegates.scrollok scrollok;
  327. public readonly Delegates.nl nl;
  328. public readonly Delegates.nonl nonl;
  329. public readonly Delegates.setscrreg setscrreg;
  330. public readonly Delegates.refresh refresh;
  331. public readonly Delegates.doupdate doupdate;
  332. public readonly Delegates.wrefresh wrefresh;
  333. public readonly Delegates.redrawwin redrawwin;
  334. //public readonly Delegates.wredrawwin wredrawwin;
  335. public readonly Delegates.wnoutrefresh wnoutrefresh;
  336. public readonly Delegates.move move;
  337. public readonly Delegates.addch addch;
  338. public readonly Delegates.addwstr addwstr;
  339. public readonly Delegates.wmove wmove;
  340. public readonly Delegates.waddch waddch;
  341. public readonly Delegates.attron attron;
  342. public readonly Delegates.attroff attroff;
  343. public readonly Delegates.attrset attrset;
  344. public readonly Delegates.getch getch;
  345. public readonly Delegates.get_wch get_wch;
  346. public readonly Delegates.ungetch ungetch;
  347. public readonly Delegates.mvgetch mvgetch;
  348. public readonly Delegates.has_colors has_colors;
  349. public readonly Delegates.start_color start_color;
  350. public readonly Delegates.init_pair init_pair;
  351. public readonly Delegates.use_default_colors use_default_colors;
  352. public readonly Delegates.COLOR_PAIRS COLOR_PAIRS;
  353. public readonly Delegates.getmouse getmouse;
  354. public readonly Delegates.ungetmouse ungetmouse;
  355. public readonly Delegates.mouseinterval mouseinterval;
  356. public readonly Delegates.mousemask mousemask;
  357. public UnmanagedLibrary UnmanagedLibrary;
  358. public NativeMethods (UnmanagedLibrary lib)
  359. {
  360. this.UnmanagedLibrary = lib;
  361. initscr = lib.GetNativeMethodDelegate<Delegates.initscr> ("initscr");
  362. endwin = lib.GetNativeMethodDelegate<Delegates.endwin> ("endwin");
  363. isendwin = lib.GetNativeMethodDelegate<Delegates.isendwin> ("isendwin");
  364. cbreak = lib.GetNativeMethodDelegate<Delegates.cbreak> ("cbreak");
  365. nocbreak = lib.GetNativeMethodDelegate<Delegates.nocbreak> ("nocbreak");
  366. echo = lib.GetNativeMethodDelegate<Delegates.echo> ("echo");
  367. noecho = lib.GetNativeMethodDelegate<Delegates.noecho> ("noecho");
  368. halfdelay = lib.GetNativeMethodDelegate<Delegates.halfdelay> ("halfdelay");
  369. raw = lib.GetNativeMethodDelegate<Delegates.raw> ("raw");
  370. noraw = lib.GetNativeMethodDelegate<Delegates.noraw> ("noraw");
  371. noqiflush = lib.GetNativeMethodDelegate<Delegates.noqiflush> ("noqiflush");
  372. qiflush = lib.GetNativeMethodDelegate<Delegates.qiflush> ("qiflush");
  373. typeahead = lib.GetNativeMethodDelegate<Delegates.typeahead> ("typeahead");
  374. timeout = lib.GetNativeMethodDelegate<Delegates.timeout> ("timeout");
  375. wtimeout = lib.GetNativeMethodDelegate<Delegates.wtimeout> ("wtimeout");
  376. notimeout = lib.GetNativeMethodDelegate<Delegates.notimeout> ("notimeout");
  377. keypad = lib.GetNativeMethodDelegate<Delegates.keypad> ("keypad");
  378. meta = lib.GetNativeMethodDelegate<Delegates.meta> ("meta");
  379. intrflush = lib.GetNativeMethodDelegate<Delegates.intrflush> ("intrflush");
  380. clearok = lib.GetNativeMethodDelegate<Delegates.clearok> ("clearok");
  381. idlok = lib.GetNativeMethodDelegate<Delegates.idlok> ("idlok");
  382. idcok = lib.GetNativeMethodDelegate<Delegates.idcok> ("idcok");
  383. immedok = lib.GetNativeMethodDelegate<Delegates.immedok> ("immedok");
  384. leaveok = lib.GetNativeMethodDelegate<Delegates.leaveok> ("leaveok");
  385. wsetscrreg = lib.GetNativeMethodDelegate<Delegates.wsetscrreg> ("wsetscrreg");
  386. scrollok = lib.GetNativeMethodDelegate<Delegates.scrollok> ("scrollok");
  387. nl = lib.GetNativeMethodDelegate<Delegates.nl> ("nl");
  388. nonl = lib.GetNativeMethodDelegate<Delegates.nonl> ("nonl");
  389. setscrreg = lib.GetNativeMethodDelegate<Delegates.setscrreg> ("setscrreg");
  390. refresh = lib.GetNativeMethodDelegate<Delegates.refresh> ("refresh");
  391. doupdate = lib.GetNativeMethodDelegate<Delegates.doupdate> ("doupdate");
  392. wrefresh = lib.GetNativeMethodDelegate<Delegates.wrefresh> ("wrefresh");
  393. redrawwin = lib.GetNativeMethodDelegate<Delegates.redrawwin> ("redrawwin");
  394. //wredrawwin = lib.GetNativeMethodDelegate<Delegates.wredrawwin> ("wredrawwin");
  395. wnoutrefresh = lib.GetNativeMethodDelegate<Delegates.wnoutrefresh> ("wnoutrefresh");
  396. move = lib.GetNativeMethodDelegate<Delegates.move> ("move");
  397. addch = lib.GetNativeMethodDelegate<Delegates.addch>("addch");
  398. addwstr = lib.GetNativeMethodDelegate<Delegates.addwstr> ("addwstr");
  399. wmove = lib.GetNativeMethodDelegate<Delegates.wmove> ("wmove");
  400. waddch = lib.GetNativeMethodDelegate<Delegates.waddch> ("waddch");
  401. attron = lib.GetNativeMethodDelegate<Delegates.attron> ("attron");
  402. attroff = lib.GetNativeMethodDelegate<Delegates.attroff> ("attroff");
  403. attrset = lib.GetNativeMethodDelegate<Delegates.attrset> ("attrset");
  404. getch = lib.GetNativeMethodDelegate<Delegates.getch> ("getch");
  405. get_wch = lib.GetNativeMethodDelegate<Delegates.get_wch> ("get_wch");
  406. ungetch = lib.GetNativeMethodDelegate<Delegates.ungetch> ("ungetch");
  407. mvgetch = lib.GetNativeMethodDelegate<Delegates.mvgetch> ("mvgetch");
  408. has_colors = lib.GetNativeMethodDelegate<Delegates.has_colors> ("has_colors");
  409. start_color = lib.GetNativeMethodDelegate<Delegates.start_color> ("start_color");
  410. init_pair = lib.GetNativeMethodDelegate<Delegates.init_pair> ("init_pair");
  411. use_default_colors = lib.GetNativeMethodDelegate<Delegates.use_default_colors> ("use_default_colors");
  412. COLOR_PAIRS = lib.GetNativeMethodDelegate<Delegates.COLOR_PAIRS> ("COLOR_PAIRS");
  413. getmouse = lib.GetNativeMethodDelegate<Delegates.getmouse> ("getmouse");
  414. ungetmouse = lib.GetNativeMethodDelegate<Delegates.ungetmouse> ("ungetmouse");
  415. mouseinterval = lib.GetNativeMethodDelegate<Delegates.mouseinterval> ("mouseinterval");
  416. mousemask = lib.GetNativeMethodDelegate<Delegates.mousemask> ("mousemask");
  417. }
  418. }
  419. }