mirror of
https://github.com/ksherlock/TwoTerm.git
synced 2025-01-02 16:31:38 +00:00
70b38025ba
git-svn-id: svn://qnap.local/TwoTerm/trunk@3154 5590a31f-7b70-45f8-8c82-aa3a8e5f4507
381 lines
8.9 KiB
Plaintext
381 lines
8.9 KiB
Plaintext
//
|
|
// TermWindowController.m
|
|
// 2Term
|
|
//
|
|
// Created by Kelvin Sherlock on 7/2/2010.
|
|
// Copyright 2010 __MyCompanyName__. All rights reserved.
|
|
//
|
|
|
|
#import "TermWindowController.h"
|
|
#import "EmulatorView.h"
|
|
#import "CurveView.h"
|
|
#import "EmulatorWindow.h"
|
|
|
|
#import "VT52.h"
|
|
#import "PTSE.h"
|
|
|
|
#import "Defaults.h"
|
|
|
|
|
|
#define TTYDEFCHARS
|
|
|
|
#include <util.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <sys/ttydefaults.h>
|
|
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
@implementation TermWindowController
|
|
|
|
@synthesize emulator = _emulator;
|
|
@synthesize emulatorView = _emulatorView;
|
|
@synthesize colorView = _colorView;
|
|
|
|
@synthesize parameters = _parameters;
|
|
|
|
+(id)new
|
|
{
|
|
return [[self alloc] initWithWindowNibName: @"TermWindow"];
|
|
}
|
|
|
|
-(void)dealloc
|
|
{
|
|
[_emulator release];
|
|
[_emulatorView release];
|
|
[_colorView release];
|
|
|
|
[_parameters release];
|
|
|
|
|
|
[super dealloc];
|
|
}
|
|
|
|
/*
|
|
-(void)awakeFromNib
|
|
{
|
|
[self initPTY];
|
|
}
|
|
*/
|
|
|
|
-(void)initPTY
|
|
{
|
|
static std::string username;
|
|
|
|
struct termios term;
|
|
struct winsize ws = [_emulator defaultSize];
|
|
|
|
memset(&term, 0, sizeof(term));
|
|
|
|
//term.c_oflag = OPOST | ONLCR;
|
|
//term.c_lflag = ECHO;
|
|
//term.c_iflag = ICRNL; // | ICANON | ECHOE | ISIG;
|
|
|
|
term.c_oflag = TTYDEF_OFLAG;
|
|
term.c_lflag = TTYDEF_LFLAG;
|
|
term.c_iflag = TTYDEF_IFLAG;
|
|
term.c_cflag = TTYDEF_CFLAG;
|
|
|
|
term.c_ispeed = term.c_ospeed = TTYDEF_SPEED;
|
|
|
|
memcpy(term.c_cc, ttydefchars, sizeof(ttydefchars));
|
|
|
|
if ([_emulator respondsToSelector: @selector(initTerm:)])
|
|
[_emulator initTerm: &term];
|
|
|
|
|
|
// getlogin() sometimes returns crap.
|
|
if (username.empty()) {
|
|
username = [NSUserName() UTF8String];
|
|
}
|
|
//NSLog(@"%@ %s %s", NSUserName(), getlogin(), getpwent()->pw_name);
|
|
_pid = forkpty(&_fd, NULL, &term, &ws);
|
|
|
|
if (_pid < 0)
|
|
{
|
|
fprintf(stderr, "forkpty failed\n");
|
|
fflush(stderr);
|
|
|
|
return;
|
|
}
|
|
if (_pid == 0)
|
|
{
|
|
|
|
std::vector<const char *> environ;
|
|
std::string s;
|
|
|
|
|
|
s.append("TERM_PROGRAM=2Term");
|
|
s.append(1, (char)0);
|
|
|
|
s.append("LANG=C");
|
|
s.append(1, (char)0);
|
|
|
|
s.append("TERM=");
|
|
s.append([_emulator termName]);
|
|
|
|
s.append(1, (char)0);
|
|
s.append(1, (char)0);
|
|
|
|
for (std::string::size_type index = 0;;)
|
|
{
|
|
environ.push_back(&s[index]);
|
|
|
|
index = s.find((char)0, index);
|
|
if (index == std::string::npos) break;
|
|
|
|
if (s[++index] == 0) break;
|
|
|
|
}
|
|
|
|
environ.push_back(NULL);
|
|
|
|
|
|
// call login -f [username]
|
|
// -p -- do NOT ignore environment.
|
|
// export TERM=...
|
|
|
|
|
|
// TODO -- option for localhost, telnet, ssh, etc.
|
|
execle("/usr/bin/login", "login", "-pf", username.c_str(), NULL, &environ[0]);
|
|
|
|
fprintf(stderr, "execle failed: %s\n", strerror(errno));
|
|
fflush(stderr);
|
|
|
|
// should not call exit.
|
|
_exit(-1);
|
|
// child
|
|
}
|
|
|
|
|
|
|
|
[_emulatorView resizeTo: iSize(ws.ws_col, ws.ws_row) animated: NO];
|
|
|
|
|
|
NSWindow *window = [self window];
|
|
|
|
if (![_emulator resizable])
|
|
{
|
|
|
|
NSUInteger mask = [window styleMask];
|
|
|
|
|
|
[window setShowsResizeIndicator: NO];
|
|
|
|
[window setStyleMask: mask & ~NSResizableWindowMask];
|
|
}
|
|
|
|
[window setMinSize: [window frame].size];
|
|
|
|
[_emulatorView setFd: _fd];
|
|
[self monitor];
|
|
|
|
/*
|
|
if (!_childMonitor)
|
|
{
|
|
_childMonitor = [ChildMonitor new];
|
|
[_childMonitor setDelegate: _emulatorView];
|
|
}
|
|
*/
|
|
}
|
|
|
|
-(void)monitor {
|
|
|
|
int fd = _fd;
|
|
int pid = _pid;
|
|
|
|
int flags;
|
|
// non-blocking io.
|
|
if (fcntl(_fd, F_GETFL, &flags) < 0) flags = 0;
|
|
fcntl(_fd, F_SETFL, flags | O_NONBLOCK);
|
|
|
|
|
|
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
|
|
|
|
_wait_source = dispatch_source_create(DISPATCH_SOURCE_TYPE_PROC,
|
|
pid, DISPATCH_PROC_EXIT, queue);
|
|
if (_wait_source)
|
|
{
|
|
|
|
dispatch_source_set_event_handler(_wait_source, ^{
|
|
|
|
int status = 0;
|
|
int ok;
|
|
for(;;) {
|
|
ok = waitpid(pid, &status, WNOHANG);
|
|
if (ok >= 0) break;
|
|
if (errno == EINTR) continue;
|
|
break;
|
|
}
|
|
_pid = 0;
|
|
//dispatch_async(dispatch_get_main_queue(), ^(){
|
|
[_emulatorView childFinished: status];
|
|
//});
|
|
|
|
dispatch_source_cancel(_wait_source);
|
|
dispatch_release(_wait_source);
|
|
_wait_source = nullptr;
|
|
});
|
|
|
|
dispatch_resume(_wait_source);
|
|
}
|
|
|
|
|
|
_read_source = dispatch_source_create(DISPATCH_SOURCE_TYPE_READ,
|
|
fd, 0, queue);
|
|
if (_read_source)
|
|
{
|
|
// Install the event handler
|
|
dispatch_source_set_event_handler(_read_source, ^{
|
|
|
|
static uint8_t sbuffer[1024];
|
|
size_t estimated = dispatch_source_get_data(_read_source);
|
|
|
|
|
|
uint8_t *buffer = estimated > sizeof(sbuffer) ? (uint8_t *)malloc(estimated) : sbuffer;
|
|
if (buffer)
|
|
{
|
|
ssize_t actual;
|
|
|
|
for (;;) {
|
|
actual = read(fd, buffer, (estimated));
|
|
if (actual < 0) {
|
|
if (errno == EINTR) continue;
|
|
|
|
if (errno == EAGAIN) {
|
|
if (buffer != sbuffer) free(buffer);
|
|
return;
|
|
}
|
|
|
|
NSLog(@"read: %s", strerror(errno));
|
|
dispatch_source_cancel(_read_source);
|
|
dispatch_release(_read_source);
|
|
_read_source = nullptr;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (actual > 0) [_emulatorView processData: buffer size: actual];
|
|
|
|
if (buffer != sbuffer) free(buffer);
|
|
|
|
if (actual == 0) {
|
|
NSLog(@"closing fd");
|
|
dispatch_source_cancel(_read_source);
|
|
dispatch_release(_read_source);
|
|
_read_source = nullptr;
|
|
}
|
|
}
|
|
});
|
|
|
|
|
|
dispatch_source_set_cancel_handler(_read_source, ^{
|
|
_fd = -1;
|
|
[_emulatorView setFd: -1];
|
|
close(fd);
|
|
});
|
|
|
|
dispatch_resume(_read_source);
|
|
}
|
|
}
|
|
|
|
#pragma mark -
|
|
#pragma mark NSWindowDelegate
|
|
|
|
- (void)windowDidLoad
|
|
{
|
|
|
|
NSColor *foregroundColor;
|
|
NSColor *backgroundColor;
|
|
Class klass;
|
|
id o;
|
|
|
|
NSWindow *window;
|
|
|
|
[super windowDidLoad];
|
|
|
|
window = [self window];
|
|
|
|
//[(CurveView *)[window contentView] setColor: [NSColor clearColor]];
|
|
|
|
//[window setContentView: _curveView];
|
|
|
|
// resize in 2.0 height increments to prevent jittering the scan lines.
|
|
//[window setResizeIncrements: NSMakeSize(1.0, 2.0)];
|
|
|
|
|
|
klass = [_parameters objectForKey: kClass];
|
|
if (!klass || ![klass conformsToProtocol: @protocol(Emulator)])
|
|
{
|
|
klass = [VT52 class];
|
|
}
|
|
|
|
o = [_parameters objectForKey: kForegroundColor];
|
|
foregroundColor = o ? (NSColor *)o : [NSColor greenColor];
|
|
|
|
o = [_parameters objectForKey: kBackgroundColor];
|
|
backgroundColor = o ? (NSColor *)o : [NSColor blackColor];
|
|
|
|
|
|
[self willChangeValueForKey: @"emulator"];
|
|
_emulator = [klass new];
|
|
[self didChangeValueForKey: @"emulator"];
|
|
|
|
[window setBackgroundColor: backgroundColor];
|
|
[(EmulatorWindow *)window setTitleTextColor: foregroundColor];
|
|
|
|
[_emulatorView setEmulator: _emulator];
|
|
[_emulatorView setForegroundColor: foregroundColor];
|
|
[_emulatorView setBackgroundColor: backgroundColor];
|
|
//[_emulatorView setScanLines: scanLines];
|
|
|
|
[_colorView setColor: backgroundColor];
|
|
|
|
o = [_parameters objectForKey: kContentFilters];
|
|
if (o)
|
|
{
|
|
[_colorView setWantsLayer: YES];
|
|
[_colorView setContentFilters: (NSArray *)o];
|
|
}
|
|
|
|
[self initPTY];
|
|
|
|
}
|
|
|
|
-(void)windowWillClose:(NSNotification *)notification
|
|
{
|
|
|
|
|
|
if (_wait_source) {
|
|
dispatch_source_cancel(_wait_source);
|
|
dispatch_release(_wait_source);
|
|
}
|
|
|
|
if (_read_source) {
|
|
dispatch_source_cancel(_read_source);
|
|
dispatch_release(_read_source);
|
|
}
|
|
|
|
int status;
|
|
int ok;
|
|
if (_pid) {
|
|
kill(_pid, 9);
|
|
for(;;) {
|
|
ok = waitpid(_pid, &status, 0);
|
|
if (ok >= 0) break;
|
|
if (errno == EINTR) continue;
|
|
perror("waitpid: ");
|
|
break;
|
|
}
|
|
}
|
|
|
|
[self autorelease];
|
|
}
|
|
|
|
@end
|
|
|
|
|
|
|