Bram Moolenaar | edf3f97 | 2016-08-29 22:49:24 +0200 | [diff] [blame] | 1 | /* vi:set ts=8 sts=4 sw=4 noet: |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 2 | * |
| 3 | * VIM - Vi IMproved by Bram Moolenaar |
| 4 | * |
| 5 | * Do ":help uganda" in Vim to read copying and usage conditions. |
| 6 | * Do ":help credits" in Vim to see a list of people who contributed. |
| 7 | */ |
| 8 | |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 9 | // Before Including the MacOS specific files, |
| 10 | // lets set the OPAQUE_TOOLBOX_STRUCTS to 0 so we |
| 11 | // can access the internal structures. |
| 12 | // (Until fully Carbon compliant) |
| 13 | // TODO: Can we remove this? (Dany) |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 14 | #if 0 |
| 15 | # define OPAQUE_TOOLBOX_STRUCTS 0 |
| 16 | #endif |
| 17 | |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 18 | // Include MAC_OS_X_VERSION_* macros |
Bram Moolenaar | 2b1b53f | 2013-11-02 21:04:38 +0100 | [diff] [blame] | 19 | #ifdef HAVE_AVAILABILITYMACROS_H |
| 20 | # include <AvailabilityMacros.h> |
| 21 | #endif |
| 22 | |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 23 | /* |
| 24 | * Macintosh machine-dependent things. |
Bram Moolenaar | ab79bcb | 2004-07-18 21:34:53 +0000 | [diff] [blame] | 25 | * |
| 26 | * Include the Mac header files, unless also compiling with X11 (the header |
| 27 | * files have many conflicts). |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 28 | */ |
Bram Moolenaar | 164fca3 | 2010-07-14 13:58:07 +0200 | [diff] [blame] | 29 | #ifdef FEAT_GUI_MAC |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 30 | # include <Quickdraw.h> // Apple calls it QuickDraw.h... |
Bram Moolenaar | ab79bcb | 2004-07-18 21:34:53 +0000 | [diff] [blame] | 31 | # include <ToolUtils.h> |
| 32 | # include <LowMem.h> |
| 33 | # include <Scrap.h> |
| 34 | # include <Sound.h> |
| 35 | # include <TextUtils.h> |
| 36 | # include <Memory.h> |
| 37 | # include <OSUtils.h> |
| 38 | # include <Files.h> |
Bram Moolenaar | 264b74f | 2019-01-24 17:18:42 +0100 | [diff] [blame] | 39 | # include <Script.h> |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 40 | #endif |
| 41 | |
| 42 | /* |
| 43 | * Unix interface |
| 44 | */ |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 45 | #if defined(__APPLE_CC__) // for Project Builder and ... |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 46 | # include <unistd.h> |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 47 | // Get stat.h or something similar. Comment: How come some OS get in in vim.h |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 48 | # include <sys/stat.h> |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 49 | // && defined(HAVE_CURSE) |
| 50 | // The curses.h from MacOS X provides by default some BACKWARD compatibility |
| 51 | // definition which can cause us problem later on. So we undefine a few of them. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 52 | # include <curses.h> |
| 53 | # undef reg |
| 54 | # undef ospeed |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 55 | // OK defined to 0 in MacOS X 10.2 curses! Remove it, we define it to be 1. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 56 | # undef OK |
| 57 | #endif |
| 58 | #include <signal.h> |
| 59 | #include <errno.h> |
| 60 | #include <stdio.h> |
| 61 | #include <stdlib.h> |
| 62 | #include <time.h> |
Bram Moolenaar | 0fd9289 | 2006-03-09 22:27:48 +0000 | [diff] [blame] | 63 | #include <dirent.h> |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 64 | |
| 65 | /* |
| 66 | * MacOS specific #define |
| 67 | */ |
| 68 | |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 69 | // This will go away when CMD_KEY fully tested |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 70 | #define USE_CMD_KEY |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 71 | // On MacOS X use the / not the : |
| 72 | // TODO: Should file such as ~/.vimrc reside instead in |
| 73 | // ~/Library/Vim or ~/Library/Preferences/org.vim.vim/ ? (Dany) |
| 74 | // When compiled under MacOS X (including CARBON version) |
| 75 | // we use the Unix File path style. Also when UNIX is defined. |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 76 | #define USE_UNIXFILENAME |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 77 | |
| 78 | |
| 79 | /* |
| 80 | * Generic Vim #define |
| 81 | */ |
| 82 | |
| 83 | #define FEAT_SOURCE_FFS |
| 84 | #define FEAT_SOURCE_FF_MAC |
| 85 | |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 86 | #define USE_EXE_NAME // to find $VIM |
| 87 | #define CASE_INSENSITIVE_FILENAME // ignore case when comparing file names |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 88 | #define SPACE_IN_FILENAME |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 89 | #define BREAKCHECK_SKIP 32 // call mch_breakcheck() each time, it's |
| 90 | // quite fast. Did I forgot to update the |
| 91 | // comment |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 92 | |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 93 | #define USE_FNAME_CASE // make ":e os_Mac.c" open the file in its |
| 94 | // original case, as "os_mac.c" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 95 | #define BINARY_FILE_IO |
| 96 | #define EOL_DEFAULT EOL_MAC |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 97 | #define HAVE_AVAIL_MEM |
| 98 | |
| 99 | #ifndef HAVE_CONFIG_H |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 100 | # define HAVE_STRING_H |
| 101 | # define HAVE_STRCSPN |
| 102 | # define HAVE_MEMSET |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 103 | # define USE_TMPNAM // use tmpnam() instead of mktemp() |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 104 | # define HAVE_FCNTL_H |
| 105 | # define HAVE_QSORT |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 106 | # define HAVE_ST_MODE // have stat.st_mode |
Bram Moolenaar | a7241f5 | 2008-06-24 20:39:31 +0000 | [diff] [blame] | 107 | # define HAVE_MATH_H |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 108 | |
| 109 | # if defined(__DATE__) && defined(__TIME__) |
| 110 | # define HAVE_DATE_TIME |
| 111 | # endif |
| 112 | # define HAVE_STRFTIME |
| 113 | #endif |
| 114 | |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 115 | /* |
| 116 | * Names for the EXRC, HELP and temporary files. |
| 117 | * Some of these may have been defined in the makefile. |
| 118 | */ |
| 119 | |
| 120 | #ifndef SYS_VIMRC_FILE |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 121 | # define SYS_VIMRC_FILE "$VIM/vimrc" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 122 | #endif |
| 123 | #ifndef SYS_GVIMRC_FILE |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 124 | # define SYS_GVIMRC_FILE "$VIM/gvimrc" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 125 | #endif |
| 126 | #ifndef SYS_MENU_FILE |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 127 | # define SYS_MENU_FILE "$VIMRUNTIME/menu.vim" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 128 | #endif |
| 129 | #ifndef SYS_OPTWIN_FILE |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 130 | # define SYS_OPTWIN_FILE "$VIMRUNTIME/optwin.vim" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 131 | #endif |
Bram Moolenaar | 8c08b5b | 2016-07-28 22:24:15 +0200 | [diff] [blame] | 132 | #ifndef VIM_DEFAULTS_FILE |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 133 | # define VIM_DEFAULTS_FILE "$VIMRUNTIME/defaults.vim" |
Bram Moolenaar | 8c08b5b | 2016-07-28 22:24:15 +0200 | [diff] [blame] | 134 | #endif |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 135 | #ifndef EVIM_FILE |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 136 | # define EVIM_FILE "$VIMRUNTIME/evim.vim" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 137 | #endif |
| 138 | |
| 139 | #ifdef FEAT_GUI |
| 140 | # ifndef USR_GVIMRC_FILE |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 141 | # define USR_GVIMRC_FILE "~/.gvimrc" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 142 | # endif |
| 143 | # ifndef GVIMRC_FILE |
| 144 | # define GVIMRC_FILE "_gvimrc" |
| 145 | # endif |
| 146 | #endif |
| 147 | #ifndef USR_VIMRC_FILE |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 148 | # define USR_VIMRC_FILE "~/.vimrc" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 149 | #endif |
| 150 | |
| 151 | #ifndef USR_EXRC_FILE |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 152 | # define USR_EXRC_FILE "~/.exrc" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 153 | #endif |
| 154 | |
| 155 | #ifndef VIMRC_FILE |
| 156 | # define VIMRC_FILE "_vimrc" |
| 157 | #endif |
| 158 | |
| 159 | #ifndef EXRC_FILE |
| 160 | # define EXRC_FILE "_exrc" |
| 161 | #endif |
| 162 | |
| 163 | #ifndef DFLT_HELPFILE |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 164 | # define DFLT_HELPFILE "$VIMRUNTIME/doc/help.txt" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 165 | #endif |
| 166 | |
| 167 | #ifndef FILETYPE_FILE |
| 168 | # define FILETYPE_FILE "filetype.vim" |
| 169 | #endif |
| 170 | #ifndef FTPLUGIN_FILE |
| 171 | # define FTPLUGIN_FILE "ftplugin.vim" |
| 172 | #endif |
| 173 | #ifndef INDENT_FILE |
| 174 | # define INDENT_FILE "indent.vim" |
| 175 | #endif |
| 176 | #ifndef FTOFF_FILE |
| 177 | # define FTOFF_FILE "ftoff.vim" |
| 178 | #endif |
| 179 | #ifndef FTPLUGOF_FILE |
| 180 | # define FTPLUGOF_FILE "ftplugof.vim" |
| 181 | #endif |
| 182 | #ifndef INDOFF_FILE |
| 183 | # define INDOFF_FILE "indoff.vim" |
| 184 | #endif |
| 185 | |
| 186 | #ifndef SYNTAX_FNAME |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 187 | # define SYNTAX_FNAME "$VIMRUNTIME/syntax/%s.vim" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 188 | #endif |
| 189 | |
| 190 | #ifdef FEAT_VIMINFO |
| 191 | # ifndef VIMINFO_FILE |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 192 | # define VIMINFO_FILE "~/.viminfo" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 193 | # endif |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 194 | #endif // FEAT_VIMINFO |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 195 | |
| 196 | #ifndef DFLT_BDIR |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 197 | # define DFLT_BDIR "." // default for 'backupdir' |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 198 | #endif |
| 199 | |
| 200 | #ifndef DFLT_DIR |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 201 | # define DFLT_DIR "." // default for 'directory' |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 202 | #endif |
| 203 | |
| 204 | #ifndef DFLT_VDIR |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 205 | # define DFLT_VDIR "$VIM/vimfiles/view" // default for 'viewdir' |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 206 | #endif |
| 207 | |
| 208 | #define DFLT_ERRORFILE "errors.err" |
Bram Moolenaar | ab79bcb | 2004-07-18 21:34:53 +0000 | [diff] [blame] | 209 | |
| 210 | #ifndef DFLT_RUNTIMEPATH |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 211 | # define DFLT_RUNTIMEPATH "~/.vim,$VIM/vimfiles,$VIMRUNTIME,$VIM/vimfiles/after,~/.vim/after" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 212 | #endif |
Bram Moolenaar | 0726870 | 2018-03-01 21:57:32 +0100 | [diff] [blame] | 213 | #ifndef CLEAN_RUNTIMEPATH |
| 214 | # define CLEAN_RUNTIMEPATH "$VIM/vimfiles,$VIMRUNTIME,$VIM/vimfiles/after" |
| 215 | #endif |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 216 | |
| 217 | /* |
| 218 | * Macintosh has plenty of memory, use large buffers |
| 219 | */ |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 220 | #define CMDBUFFSIZE 1024 // size of the command processing buffer |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 221 | |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 222 | #ifndef DFLT_MAXMEM |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 223 | # define DFLT_MAXMEM 512 // use up to 512 Kbyte for buffer |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 224 | #endif |
| 225 | |
| 226 | #ifndef DFLT_MAXMEMTOT |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 227 | # define DFLT_MAXMEMTOT 2048 // use up to 2048 Kbyte for Vim |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 228 | #endif |
| 229 | |
| 230 | #define WILDCHAR_LIST "*?[{`$" |
| 231 | |
| 232 | /**************/ |
| 233 | #define mch_rename(src, dst) rename(src, dst) |
| 234 | #define mch_remove(x) unlink((char *)(x)) |
| 235 | #ifndef mch_getenv |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 236 | # if defined(__APPLE_CC__) |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 237 | # define mch_getenv(name) ((char_u *)getenv((char *)(name))) |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 238 | # define mch_setenv(name, val, x) setenv(name, val, x) |
| 239 | # else |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 240 | // vim_getenv() is in pty.c |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 241 | # define USE_VIMPTY_GETENV |
| 242 | # define mch_getenv(x) vimpty_getenv(x) |
| 243 | # define mch_setenv(name, val, x) setenv(name, val, x) |
| 244 | # endif |
| 245 | #endif |
| 246 | |
| 247 | #ifndef HAVE_CONFIG_H |
| 248 | # ifdef __APPLE_CC__ |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 249 | // Assuming compiling for MacOS X |
| 250 | // Trying to take advantage of the prebinding |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 251 | # define HAVE_TGETENT |
| 252 | # define OSPEED_EXTERN |
| 253 | # define UP_BC_PC_EXTERN |
| 254 | # endif |
| 255 | #endif |
| 256 | |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 257 | // Some "prep work" definition to be able to compile the MacOS X |
| 258 | // version with os_unix.c instead of os_mac.c. Based on the result |
| 259 | // of ./configure for console MacOS X. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 260 | |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 261 | #ifndef SIGPROTOARG |
| 262 | # define SIGPROTOARG (int) |
| 263 | #endif |
| 264 | #ifndef SIGDEFARG |
| 265 | # define SIGDEFARG(s) (s) int s UNUSED; |
| 266 | #endif |
| 267 | #ifndef SIGDUMMYARG |
| 268 | # define SIGDUMMYARG 0 |
| 269 | #endif |
| 270 | #undef HAVE_AVAIL_MEM |
| 271 | #ifndef HAVE_CONFIG_H |
| 272 | # define RETSIGTYPE void |
| 273 | # define SIGRETURN return |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 274 | //# define USE_SYSTEM // Output ship do debugger :(, but not compile |
| 275 | # define HAVE_SYS_WAIT_H 1 // Attempt |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 276 | # define HAVE_TERMIOS_H 1 |
| 277 | # define SYS_SELECT_WITH_SYS_TIME 1 |
| 278 | # define HAVE_SELECT 1 |
| 279 | # define HAVE_SYS_SELECT_H 1 |
| 280 | # define HAVE_PUTENV |
| 281 | # define HAVE_SETENV |
| 282 | # define HAVE_RENAME |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 283 | #endif |
| 284 | |
Bram Moolenaar | d057301 | 2017-10-28 21:11:06 +0200 | [diff] [blame] | 285 | #if !defined(HAVE_CONFIG_H) |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 286 | # define HAVE_PUTENV |
| 287 | #endif |
| 288 | |
Bram Moolenaar | 9bf703d | 2019-11-30 19:44:38 +0100 | [diff] [blame] | 289 | // A Mac constant causing big problem to syntax highlighting |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 290 | #define UNKNOWN_CREATOR '\?\?\?\?' |