Xwindows和一些编码

 

Xwindows--X server:
一个程序,它在某台机器上运行,接受远方clinet的请求,画出请求的图形。并把得到的响应发送给client,以便使得client知道下一步的动作,并把相应信息返还给server,以便更新图形。
 
本机telnet 启动服务器sun服务器,sun服务器的监听进程fork出telnet进程服务请求的机子.telnet进程fork出来两个,父进程负责与远方的机子通信,子进程exec成shell进程,解释执行父进程得到的命令,再将结果返回给远方的机子。
 
根据Xwindows原理,远方的机子应该是server,将收到的服务器的结果显示给用户看,Xwindows 的网络透明,本机telnet到多个远发的服务器,他们之间可以相互拷贝,使用鼠标选中一段字符,然后就可以在另外的程序里把该段字符粘贴过去,不论程序运行在什么地方。
 
窗口管理器:
窗口管理器是Gnome和KDE的一部分,主要功能:改变窗口大小,移动窗口,改变窗口的层叠顺序。
通常的x客户程序不需要知道有人想移动它,它只听窗口管理器的。你的窗口上的标题,按钮,全部都是窗口管理器提供的。关闭一个程序的窗口,是窗口管理器通知那个程序“有人要关掉你,你自己准备后事,然后退出吧”
 
系统的编码的问题在linux下经常遇到,例如windows下面的文件有时候名字和内容在linux下面是乱码,这主要是和linux的utf8编码与windows的gbk不同所导致的。在linux下面创建的文件默认是utf8,所以当用u盘拷贝到别人的windows机器时就变成了乱码。同理windows下面的问题拷贝到linux下面的问题,这可以通过mount 的时候增加转码的参数解决-o iocharset=utf8就ok了,这里的utf8是本地编码,可以通过locale命令实现

Latex xeCJK安装

最近在小某某的感召下用起了shell编程,进而拷来了texlive,中文支持的时候出了些小小的问题,现在分享下简单的解决方法。

先是利用安装盘里的texlive的包install.sh到本地硬盘,做得貌似就是复制i386的工作?

简单的英文文档支持没有问题,为了支持中文

 

cp -r /usr/local/texlive/2012/texmf-dist/tex/xelatex/xecjk  /usr/local/texlive/texmf-local/tex/xelatex/ 中,

注意root权限

 

利用texlive中的命令/usr/local/texlive/2012/bin/i386-linux/mktexlsr 更新包信息

 

简单的测试程序如下:

1.tex

 

\documentclass[12pt]{article}
\usepackage{xeCJK}
\setCJKmainfont{AR PL UKai CN}
\begin{document}
这是测试要显示的内容
\end{document}
 
利用xelatex命令
xelatex 2.tex 生成2.pdf

关于C语言的函数调用机制

今天在sina Weibo上看到一个人发的一道面试题目,有关于被调用函数中局部变量的若干问题。于是我回忆函数的调用机制,但是感觉有点模糊,以前在做MIT的JOS的时候碰到过,但是记忆有些模糊了,于是利用一个简单的程序进行了过程的重温。

OS是gentoo,GCC-4.5.3--gcc -S  *.c 进行反汇编

 

源代码如下

 

#include <stdio.h>

struct tree{
int a;
int b;
};//之所以定义结构体是想更深入的了解下
struct tree foo(int a, char b)
{
   struct tree m = {1,2};
   m.a = a;
   m.b = b;
   return m;
}

int main(void)
{
    int a = 2;
    char b = 3;
    struct tree m = {4,4}; 
    m = foo(a, b);
    a = m.a;
}
 
gcc反汇编后的代码:
	.file	"1.c"
	.text
.globl foo
	.type	foo, @function
foo:
	pushl	%ebp
	movl	%esp, %ebp
	subl	$20, %esp
	movl	8(%ebp), %ecx
	movl	16(%ebp), %eax
	movb	%al, -20(%ebp)
	movl	$1, -8(%ebp)
	movl	$2, -4(%ebp)
	movl	12(%ebp), %eax
	movl	%eax, -8(%ebp)
	movsbl	-20(%ebp), %eax
	movl	%eax, -4(%ebp)
	movl	-8(%ebp), %eax
	movl	-4(%ebp), %edx
	movl	%eax, (%ecx)
	movl	%edx, 4(%ecx)
	movl	%ecx, %eax
	leave
	ret	$4
	.size	foo, .-foo
.globl main
	.type	main, @function
main:
	pushl	%ebp
	movl	%esp, %ebp
	subl	$28, %esp
	movl	$2, -4(%ebp)
	movb	$3, -5(%ebp)
	movl	$4, -16(%ebp)
	movl	$4, -12(%ebp)
	movsbl	-5(%ebp), %edx
	leal	-16(%ebp), %eax
	movl	%edx, 8(%esp)
	movl	-4(%ebp), %edx
	movl	%edx, 4(%esp)
	movl	%eax, (%esp)
	call	foo
	subl	$4, %esp
	movl	-16(%ebp), %eax
	movl	%eax, -4(%ebp)
	leave
	ret
	.size	main, .-main
	.ident	"GCC: (Gentoo 4.5.3-r2 p1.5, pie-0.4.7) 4.5.3"
	.section	.note.GNU-stack,"",@progbits

 

 

其中指令比较有难度的就是

call 函数调用指令,在main函数中我们调用了foo函数,对应的汇编指令就是

call foo,操作系统做的工作就是sp=sp-4,然后将ip入栈,更换ip指向foo的指令

 

leave,将ebp的值赋给esp,然后pop出来ebp。注意pop 指令先读取出esp 对应栈单元的值,然后esp=esp+4;push指令先esp=esp-4,在将值存入esp所指的栈单元。

 

ret,函数返回指令,根据当前的esp读取出eip的值,然后返回主函数的指令

 

下面根据汇编代码和内存栈的变化说明程序的执行过程以及函数调用如何得到返回值

执行到调用call foo时内存栈的情况如下

注意leal    -16(%ebp), %eax ;将ebp-16的值赋给eax,通过记录结构体内存单元的地址实现返回值的使用。

 

call指令执行过后,执行到处理返回值之前,内存单元的情况如:

下面是函数是如何将返回值传递给主函数的

通过ebp,esp的变换,在调用函数中实现了返回值的传递。

 

下面执行到leave指令,复原esp,ebp,然后ret返回到main函数中。

上诉实验说明了很多东西以前貌似懂了,但是讲的话难度还是很大。反正赋闲在家,有心情的话还想把JOS重新做一边,顺便把很多书上的东西巩固一下。

VIM之cscope和taglist

一直想尝试下放弃sourceinsight,今天中午终于有时间试了下cscope和taglist插件

cscope主要是生成文档之间和函数之间的以来关系,比ctag要强大,taglsit主要是显示像sourceinsight一样的声明信息

 

cscope可以通过yum install安装

taglist主要是vim的插件,官网上可以下载,放到~/.vim/下(如果没有,则创建,记得打开vim的后,输入helptags  ~/.vim/doc/ ,这样才能查看 help文档)

 

在工程的根目录下面使用cscope的命令

find -name "*.h" -o -name "*.c" -o -name "*.cpp" > cscope.file

cscope -bkq -i cscope.file

ctags -R

分别查找到工程目录文件,添加到cscope.file中,然后根据它里面的内容,生成依赖关系

 

taglist的配置,可以写到/etc/vimrc 里面

let Tlist_Ctags_Cmd = '/usr/bin/ctags'

let Tlist_Show_One_File = 1

let Tlist_Exit_OnlyWindow = 1         仅标签页存在则关闭

 

 

在命令模式输入TlistOpen打开

ctrl+w可以在左右窗口切换

ctrl+t生成显示该定义的新的标签页,命令行模式输入tabn,tabp可切换

 

 

 

autoscan autoconf自动生成 makefile

在lab文件夹下面建了三个.cpp文件

main.cpp 调用了check.cpp, getinfo.cpp里面的自定义函数

 

1.首先执行autoscan

将生成的configure.scan改名为configure.in

 

修改后的为

 

#                                               -*- Autoconf -*-
# Process this file with autoconf to produce a configure script.
 
AC_PREREQ([2.65])
AC_INIT(AI, 1.0, zhaoxiahust@gmail.com)
AC_CONFIG_SRCDIR([getinfo.cpp])
AC_CONFIG_HEADERS([config.h])
AM_INIT_AUTOMAKE(AI,1.0)
# Checks for programs.
AC_PROG_CXX
 
# Checks for libraries.
 
# Checks for header files.
 
# Checks for typedefs, structures, and compiler characteristics.
 
# Checks for library functions.
 
AC_OUTPUT([Makefile])
 
其中需要注意的是AC_PROG_CXX为c++编译器
如果是.c文件则为AC_PROG_CC
 
 
 
2.运行aclocal生成一个“aclocal.m4”文件和一个缓冲文件夹autom4te.cache,该文件主要处理本地的宏定义。
3.运行autoconf,生成configure
 
4.运行 autoheader,它负责生成config.h.in文件。该工具通常会从“acconfig.h”文件中复制用户附加的符号定义,因此此处没有附加符号定义,所以不需要创建“acconfig.h”文件。
 
创建一个 Makefile.am.这一步是创建Makefile很重要的一步,automake要用的脚本配置文件是Makefile.am,用户需要自己创建相应的文件。之后,automake工具转换成Makefile.in。
 
 
Makefile.am内容
AUTOMAKE_OPTIONS=foreign
bin_PROGRAMS=AI
AI_SOURCES=getinfo.cpp main.cpp check.cpp
 
 
使用automake对其生成“configure.in”文件,在这里使用选项“—add-missing”可以让automake自动添加有一些必需的脚本文件。
 
5.运行automake --add-missing
6.运行./configure,在这一步中,通过运行自动配置设置文件configure,把Makefile.in变成了最终的Makefile。
7.make
生成可执行程序AI
 
OVER

 

 

 

 

 

补充:

今天又将文件分在了source和include目录虾米,唯一要改的就是AI_SOURCES=source/getinfo.cpp source/main.cpp source/check.cpp

这个地方

 

因为又涉及到了简单多线程编程,所以要用到pthread库,但是由于不是linux的默认链接库,所以要在Makefile.am中添加

AI_LDFLAGS=-lpthread

 

 

 

_LDFLAGS:链接选项,对应gcc命令的-L,-I,-shared等选项

 

 

 

还要在configure.in中添加

 

# Checks for libraries.
AC_CHECK_LIB([pthread], [main])