summaryrefslogtreecommitdiff
path: root/extensions/source/nsplugin/source/so_main.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'extensions/source/nsplugin/source/so_main.cxx')
-rw-r--r--extensions/source/nsplugin/source/so_main.cxx499
1 files changed, 499 insertions, 0 deletions
diff --git a/extensions/source/nsplugin/source/so_main.cxx b/extensions/source/nsplugin/source/so_main.cxx
new file mode 100644
index 000000000000..ccd879020ec1
--- /dev/null
+++ b/extensions/source/nsplugin/source/so_main.cxx
@@ -0,0 +1,499 @@
+/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
+/*************************************************************************
+ *
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * Copyright 2000, 2010 Oracle and/or its affiliates.
+ *
+ * OpenOffice.org - a multi-platform office productivity suite
+ *
+ * This file is part of OpenOffice.org.
+ *
+ * OpenOffice.org is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License version 3
+ * only, as published by the Free Software Foundation.
+ *
+ * OpenOffice.org is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU Lesser General Public License version 3 for more details
+ * (a copy is included in the LICENSE file that accompanied this code).
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * version 3 along with OpenOffice.org. If not, see
+ * <http://www.openoffice.org/license.html>
+ * for a copy of the LGPLv3 License.
+ *
+ ************************************************************************/
+
+// MARKER(update_precomp.py): autogen include statement, do not remove
+#include "precompiled_extensions.hxx"
+#ifdef UNIX
+#include <sys/stat.h>
+#include <sys/socket.h>
+#include <sys/types.h>
+#include <arpa/inet.h>
+#include <netinet/in.h>
+#endif //end of UNIX
+
+#ifdef WNT
+#ifdef _MSC_VER
+#pragma once
+#endif
+#pragma warning (push,1)
+#pragma warning (disable:4668)
+#define WIN32_LEAN_AND_MEAN
+#include <windows.h>
+#include <winsock2.h>
+#include <malloc.h>
+#include <memory.h>
+#include <tchar.h>
+#pragma warning (pop)
+#endif //end of WNT
+
+#include <unistd.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <errno.h>
+#include "boost/scoped_array.hpp"
+
+#include "ns_debug.hxx"
+#include "so_msg.hxx"
+#include "so_instance.hxx"
+#include "so_env.hxx"
+
+#include "nsp_func.hxx"
+
+#include "sal/main.h"
+#include <sal/macros.h>
+
+#include "rtl/process.h"
+#include "rtl/bootstrap.hxx"
+#include "rtl/string.hxx"
+#include "rtl/ustrbuf.hxx"
+
+#include "osl/security.hxx"
+#include "osl/thread.hxx"
+
+#include "cppuhelper/bootstrap.hxx"
+
+
+
+#include "com/sun/star/uno/XComponentContext.hpp"
+#include "com/sun/star/lang/XMultiServiceFactory.hpp"
+#include "com/sun/star/bridge/UnoUrlResolver.hpp"
+#include "com/sun/star/bridge/XUnoUrlResolver.hpp"
+
+#define OUSTR(x) ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM(x) )
+
+using namespace ::rtl;
+using namespace ::osl;
+using namespace ::cppu;
+using namespace ::com::sun::star;
+using namespace ::com::sun::star::uno;
+
+
+#define MAX_NODE_NUM 1024
+
+SoPluginInstance* lpInstance[MAX_NODE_NUM];
+
+static NSP_PIPE_FD la_read_fd = 0;
+static char const * progdir = NULL;
+
+
+long int NSP_ReadFromPipe(NSP_PIPE_FD fp, void* buf, unsigned long int len)
+{
+ unsigned long int len_unix = 0, len_wnt = 0;
+
+ len_unix = NSP_Read_Pipe(fp, buf, len, &len_wnt);
+#ifdef UNIX
+ (void)len_wnt;
+ return len_unix;
+#endif //end of UNIX
+#ifdef WNT
+ return len_wnt;
+#endif //end of WNT
+
+}
+
+int find_free_node()
+{
+ for(int i=0; i<MAX_NODE_NUM; i++)
+ {
+ if(NULL == lpInstance[i])
+ return i;
+ }
+ return -1;
+}
+
+int find_cur_node(long cur_id)
+{
+ for(int i=0; i<MAX_NODE_NUM; i++)
+ {
+ if(lpInstance[i] == NULL)
+ continue;
+ if(cur_id == lpInstance[i]->GetParent())
+ return i;
+ }
+ return -1;
+}
+
+sal_Bool dump_plugin_message(PLUGIN_MSG* pMsg)
+{
+ if (!pMsg)
+ return sal_False;
+ debug_fprintf(NSP_LOG_APPEND, "NSPlugin Message: msg_id:%d; instance_id:%d;wnd_id:%d;wnd_x:%d;wnd_y:%d;wnd_w:%d;wnd_h:%d; url:%s\n",
+ pMsg->msg_id, pMsg->instance_id, pMsg->wnd_id,
+ pMsg->wnd_x, pMsg->wnd_y, pMsg->wnd_w, pMsg->wnd_h, pMsg->url);
+ return sal_True;
+}
+
+int Set_Window(PLUGIN_MSG* pMsg)
+{
+ dump_plugin_message(pMsg);
+ int cur_no;
+ if( -1 == (cur_no = find_cur_node(pMsg->instance_id)))
+ return -1;
+ if(lpInstance[cur_no]->SetWindow(pMsg->wnd_id,
+ pMsg->wnd_x, pMsg->wnd_y, pMsg->wnd_w, pMsg->wnd_h))
+ return 0;
+ else
+ return -1;
+}
+
+int Set_URL(PLUGIN_MSG* pMsg)
+{
+ dump_plugin_message(pMsg);
+ int cur_no;
+ if( -1 == (cur_no = find_cur_node(pMsg->instance_id)))
+ return -1;
+ if(lpInstance[cur_no]->SetURL(pMsg->url))
+ return 0;
+ else
+ return -1;
+}
+
+int New_Instance(PLUGIN_MSG* pMsg, Reference< lang::XMultiServiceFactory > xMSF)
+{
+ dump_plugin_message(pMsg);
+ int free_no;
+ if( -1 == (free_no = find_free_node()))
+ return -1;
+ lpInstance[free_no] = new SoPluginInstance(pMsg->instance_id, xMSF);
+ return 0;
+}
+
+int Destroy(PLUGIN_MSG* pMsg)
+{
+ dump_plugin_message(pMsg);
+ int cur_no;
+ if( -1 == (cur_no = find_cur_node(pMsg->instance_id)))
+ return -1;
+ if(lpInstance[cur_no] != NULL)
+ {
+ lpInstance[cur_no]->Destroy();
+ debug_fprintf(NSP_LOG_APPEND, "print by Nsplugin, begin delete.\n");
+ delete(lpInstance[cur_no]);
+ lpInstance[cur_no] = NULL;
+ }
+ return 0;
+}
+
+int Print(PLUGIN_MSG* pMsg)
+{
+ dump_plugin_message(pMsg);
+ int cur_no;
+ if( -1 == (cur_no = find_cur_node(pMsg->instance_id)))
+ return -1;
+ if(lpInstance[cur_no] != NULL)
+ {
+ lpInstance[cur_no]->Print();
+ }
+ return 0;
+}
+
+int Shutdown()
+{
+ for(int cur_no=0; cur_no<MAX_NODE_NUM; cur_no++)
+ {
+ if(lpInstance[cur_no] == NULL)
+ continue;
+ lpInstance[cur_no]->Destroy();
+ debug_fprintf(NSP_LOG_APPEND, "print by Nsplugin, begin delete.\n");
+ delete(lpInstance[cur_no]);
+ lpInstance[cur_no] = NULL;
+ }
+ return -1;
+}
+
+int dispatchMsg(PLUGIN_MSG* pMsg, Reference< lang::XMultiServiceFactory > xMSF)
+{
+ switch(pMsg->msg_id)
+ {
+ case SO_SET_WINDOW:
+ return Set_Window(pMsg);
+ case SO_NEW_INSTANCE:
+ if(xMSF.is())
+ return New_Instance(pMsg, xMSF);
+ case SO_SET_URL:
+ return Set_URL(pMsg);
+ case SO_DESTROY:
+ return Destroy(pMsg);
+ case SO_SHUTDOWN:
+ Shutdown();
+ return -1;
+ case SO_PRINT:
+ Print(pMsg);
+ return 0;
+ default:
+ return -1;
+ }
+}
+
+Reference< lang::XMultiServiceFactory > SAL_CALL start_office(NSP_PIPE_FD read_fd)
+{
+ Reference< XComponentContext > xRemoteContext;
+
+ try
+ {
+ OUString aOfficePath;
+
+#ifdef UNIX
+ boost::scoped_array< char > exepath(
+ new char[( progdir ? strlen( progdir ) : 0 ) + RTL_CONSTASCII_LENGTH( "/soffice" ) + 1] );
+ if ( progdir )
+ sprintf( exepath.get(), "%s/soffice", progdir );
+ else
+ sprintf( exepath.get(), "soffice" );
+ if (!rtl_convertStringToUString(
+ &aOfficePath.pData, exepath.get(), strlen(exepath.get()), osl_getThreadTextEncoding(),
+ (RTL_TEXTTOUNICODE_FLAGS_UNDEFINED_ERROR |
+ RTL_TEXTTOUNICODE_FLAGS_MBUNDEFINED_ERROR |
+ RTL_TEXTTOUNICODE_FLAGS_INVALID_ERROR)))
+ {
+ debug_fprintf(NSP_LOG_APPEND,"bad characters in soffice installation path!\n");
+ return Reference< lang::XMultiServiceFactory >(NULL);
+ }
+#endif //end of UNIX
+#ifdef WNT
+ char sPath[NPP_PATH_MAX];
+ sPath[0] = 0;
+
+ // The quotes will be added in osl_executeProcess
+ sprintf(sPath, "%s", findSofficeExecutable() );
+ if (!rtl_convertStringToUString(
+ &aOfficePath.pData, sPath, strlen(sPath), osl_getThreadTextEncoding(),
+ (RTL_TEXTTOUNICODE_FLAGS_UNDEFINED_ERROR |
+ RTL_TEXTTOUNICODE_FLAGS_MBUNDEFINED_ERROR |
+ RTL_TEXTTOUNICODE_FLAGS_INVALID_ERROR)))
+ {
+ debug_fprintf(NSP_LOG_APPEND,"bad characters in soffice installation path!\n");
+ return Reference< lang::XMultiServiceFactory >(NULL);
+ }
+#endif //end of WNT
+
+ // create default local component context
+ Reference< XComponentContext > xLocalContext(
+ defaultBootstrap_InitialComponentContext() );
+ if ( !xLocalContext.is() )
+ {
+ debug_fprintf(NSP_LOG_APPEND,"no local component context!\n");
+ return Reference< lang::XMultiServiceFactory >(NULL);
+ }
+
+ // env string
+ ::rtl::OUStringBuffer buf;
+ OUString aIniPath, aPluginPipeName;
+
+ if(!Bootstrap::get(OUSTR("BRAND_BASE_DIR"), aIniPath))
+ {
+ debug_fprintf(NSP_LOG_APPEND,"failed to get BRAND_BASE_DIR!\n");
+ return Reference< lang::XMultiServiceFactory >(NULL);
+ }
+ aIniPath += OUSTR("/program/");
+ aIniPath += OUSTR(SAL_CONFIGFILE("bootstrap"));
+ Bootstrap aVersionFile(aIniPath);
+ aVersionFile.getFrom(OUSTR("BaseInstallation"), aPluginPipeName, OUString());
+
+ aPluginPipeName = ::rtl::OUString::valueOf( aPluginPipeName.hashCode() );
+
+ // accept string
+ OSL_ASSERT( buf.getLength() == 0 );
+ buf.appendAscii( RTL_CONSTASCII_STRINGPARAM( "-accept=pipe,name=" ) );
+ buf.append( aPluginPipeName ); //user installation path as pipe name
+ buf.appendAscii( RTL_CONSTASCII_STRINGPARAM( ";urp;" ) );
+ OUString sConnectStartString( buf.makeStringAndClear() );
+
+ // arguments
+ OUString args [] = {
+ OUSTR( "-nologo" ),
+ OUSTR( "-nodefault" ),
+ OUSTR( "-nolockcheck" ),
+ sConnectStartString,
+ };
+
+ // create a URL resolver
+ Reference< bridge::XUnoUrlResolver > xUrlResolver(
+ bridge::UnoUrlResolver::create( xLocalContext ) );
+
+ // connection string
+ OSL_ASSERT( buf.getLength() == 0 );
+ buf.appendAscii( RTL_CONSTASCII_STRINGPARAM( "uno:pipe,name=" ) );
+ buf.append( aPluginPipeName );
+ buf.appendAscii( RTL_CONSTASCII_STRINGPARAM(
+ ";urp;StarOffice.ComponentContext" ) );
+ OUString sConnectString( buf.makeStringAndClear() );
+
+ try
+ {
+ // try to connect to office, no need to start instance again if office already started
+ xRemoteContext.set(
+ xUrlResolver->resolve( sConnectString ), UNO_QUERY_THROW );
+ debug_fprintf(NSP_LOG_APPEND, "Staroffice already start\n");
+ return Reference< lang::XMultiServiceFactory >(xRemoteContext->getServiceManager(), UNO_QUERY);
+ }
+ catch ( connection::NoConnectException & )
+ {
+ }
+
+ // start office process
+#ifdef UNIX
+ // a temporary solution
+ // in future the process should be started using the osl_executeProcess call
+ int nChildPID = fork();
+ if( ! nChildPID ) // child process
+ {
+ NSP_Close_Pipe(read_fd);
+ execl( "/bin/sh",
+ "/bin/sh",
+ ::rtl::OUStringToOString( aOfficePath, osl_getThreadTextEncoding() ).getStr(),
+ ::rtl::OUStringToOString( args[0], osl_getThreadTextEncoding() ).getStr(),
+ ::rtl::OUStringToOString( args[1], osl_getThreadTextEncoding() ).getStr(),
+ ::rtl::OUStringToOString( args[2], osl_getThreadTextEncoding() ).getStr(),
+ ::rtl::OUStringToOString( args[3], osl_getThreadTextEncoding() ).getStr(),
+ NULL);
+ _exit(255);
+ }
+#else
+ (void) read_fd; /* avoid warning about unused parameter */
+ Security sec;
+ oslProcess hProcess = 0;
+ rtl_uString * ar_args [] = {
+ args[ 0 ].pData,
+ args[ 1 ].pData,
+ args[ 2 ].pData,
+ args[ 3 ].pData,
+ };
+
+ oslProcessError rc = osl_executeProcess(
+ aOfficePath.pData,
+ ar_args,
+ SAL_N_ELEMENTS( ar_args ),
+ osl_Process_DETACHED,
+ sec.getHandle(),
+ 0, // => current working dir
+ 0,
+ 0, // => no env vars
+ &hProcess );
+ switch ( rc )
+ {
+ case osl_Process_E_None:
+ osl_freeProcessHandle( hProcess );
+ break;
+ default:
+ debug_fprintf(NSP_LOG_APPEND, "unmapped error!\n");
+ return Reference< lang::XMultiServiceFactory >(NULL);
+ }
+#endif
+
+ // wait until office is started
+ for ( int i = 0; i < 240 /* stop the connection after 240 * 500ms */; ++i )
+ {
+ try
+ {
+ // try to connect to office
+ xRemoteContext.set(
+ xUrlResolver->resolve( sConnectString ), UNO_QUERY_THROW );
+ return Reference< lang::XMultiServiceFactory >(xRemoteContext->getServiceManager(), UNO_QUERY);
+ }
+ catch ( connection::NoConnectException & )
+ {
+ // wait 500 ms, then try to connect again
+ TimeValue tv = { 0 /* secs */, 500000000 /* nanosecs */ };
+ ::osl::Thread::wait( tv );
+ }
+ }
+ debug_fprintf(NSP_LOG_APPEND, "Failed to connect to Staroffice in 2 minutes\n");
+ return Reference< lang::XMultiServiceFactory >(NULL);
+ }
+ catch ( Exception & e)
+ {
+ debug_fprintf(NSP_LOG_APPEND, "unexpected UNO exception caught: ");
+ debug_fprintf(NSP_LOG_APPEND, (sal_Char *)e.Message.getStr());
+ return Reference< lang::XMultiServiceFactory >(NULL);
+ }
+
+}
+
+
+SAL_IMPLEMENT_MAIN_WITH_ARGS(argc, argv)
+{
+ // Sleep(20*1000);
+ debug_fprintf(NSP_LOG_APPEND, "start of main\n");
+ memset(lpInstance, 0, sizeof(lpInstance));
+
+ // MessageBox( NULL, "nsplugin has been started", "Info", MB_OK );
+
+ NSP_PIPE_FD fd_pipe[2];
+ int iPipe[2];
+ if(argc < 3)
+ {
+ debug_fprintf(NSP_LOG_APPEND, "print by nsplugin, command error; too little argument to start plugin exec\n");
+ return EXIT_FAILURE;
+ }
+ iPipe[0] = atoi(argv[1]);
+ iPipe[1] = atoi(argv[2]);
+
+ // fd_pipe[0]: read, fd_pipe[0]: write
+ fd_pipe[0] = (NSP_PIPE_FD) iPipe[0] ;
+ fd_pipe[1] = (NSP_PIPE_FD) iPipe[1] ;
+ NSP_Close_Pipe(fd_pipe[1]);
+
+ la_read_fd = fd_pipe[0];
+ if(la_read_fd < 0)
+ {
+ debug_fprintf(NSP_LOG_APPEND, "print by nsplugin, command error: bad read file id:%s \n", la_read_fd);
+ return 0;
+ }
+
+ // the program path is provided only on unix, on windows the registry entry is used
+ if ( argc > 4 )
+ progdir = argv[4];
+
+ Reference< lang::XMultiServiceFactory > xFactory = start_office(la_read_fd);
+ if(!xFactory.is())
+ {
+ NSP_Close_Pipe(la_read_fd);
+ return -1;
+ }
+ PLUGIN_MSG nMsg;
+ int len;
+ while(1)
+ {
+ memset(&nMsg, 0, sizeof(PLUGIN_MSG));
+ len = NSP_ReadFromPipe(la_read_fd, (char*)&nMsg, sizeof(PLUGIN_MSG));
+ if(len != sizeof(PLUGIN_MSG))
+ break;
+ debug_fprintf(NSP_LOG_APPEND, "Read message from pipe type %d \n", nMsg.msg_id);
+ if(-1 == dispatchMsg(&nMsg, xFactory))
+ {
+ debug_fprintf(NSP_LOG_APPEND, "plugin will shutdown\n");
+ break;
+ }
+ }
+ NSP_Close_Pipe(la_read_fd);
+ _exit(0);
+ return EXIT_SUCCESS; // avoid warnings
+}
+
+/* vim:set shiftwidth=4 softtabstop=4 expandtab: */