fbcon.c

来自「优龙2410linux2.6.8内核源代码」· C语言 代码 · 共 2,412 行 · 第 1/5 页

C
2,412
字号
				attr = c & 0xff00;				if (s > start) {					accel_putcs(vc, info, start, s - start,						    real_y(p, line), x);					x += s - start;					start = s;				}			}			if (c == scr_readw(d)) {				if (s > start) {					accel_putcs(vc, info, start, s - start,						    real_y(p, line), x);					x += s - start + 1;					start = s + 1;				} else {					x++;					start++;				}			}			scr_writew(c, d);			console_conditional_schedule();			s++;			d++;		} while (s < le);		if (s > start)			accel_putcs(vc, info, start, s - start,				    real_y(p, line), x);		console_conditional_schedule();		if (offset > 0)			line++;		else {			line--;			/* NOTE: We subtract two lines from these pointers */			s -= vc->vc_size_row;			d -= vc->vc_size_row;		}	}}static inline void fbcon_softback_note(struct vc_data *vc, int t,				       int count){	unsigned short *p;	if (vc->vc_num != fg_console)		return;	p = (unsigned short *) (vc->vc_origin + t * vc->vc_size_row);	while (count) {		scr_memcpyw((u16 *) softback_in, p, vc->vc_size_row);		count--;		p = advance_row(p, 1);		softback_in += vc->vc_size_row;		if (softback_in == softback_end)			softback_in = softback_buf;		if (softback_in == softback_top) {			softback_top += vc->vc_size_row;			if (softback_top == softback_end)				softback_top = softback_buf;		}	}	softback_curr = softback_in;}static int fbcon_scroll(struct vc_data *vc, int t, int b, int dir,			int count){	struct fb_info *info = registered_fb[(int) con2fb_map[vc->vc_num]];	struct display *p = &fb_display[vc->vc_num];	int scroll_partial = info->flags & FBINFO_PARTIAL_PAN_OK;	if (!info->fbops->fb_blank && console_blanked)		return 0;	if (!count || vt_cons[vc->vc_num]->vc_mode != KD_TEXT)		return 0;	fbcon_cursor(vc, CM_ERASE);	/*	 * ++Geert: Only use ywrap/ypan if the console is in text mode	 * ++Andrew: Only use ypan on hardware text mode when scrolling the	 *           whole screen (prevents flicker).	 */	switch (dir) {	case SM_UP:		if (count > vc->vc_rows)	/* Maximum realistic size */			count = vc->vc_rows;		if (softback_top)			fbcon_softback_note(vc, t, count);		if (logo_shown >= 0)			goto redraw_up;		switch (p->scrollmode) {		case SCROLL_MOVE:			accel_bmove(vc, info, t + count, 0, t, 0,					 b - t - count, vc->vc_cols);			accel_clear(vc, info, b - count, 0, count,					 vc->vc_cols);			break;		case SCROLL_WRAP_MOVE:			if (b - t - count > 3 * vc->vc_rows >> 2) {				if (t > 0)					fbcon_bmove(vc, 0, 0, count, 0, t,						    vc->vc_cols);				ywrap_up(vc, count);				if (vc->vc_rows - b > 0)					fbcon_bmove(vc, b - count, 0, b, 0,						    vc->vc_rows - b,						    vc->vc_cols);			} else if (info->flags & FBINFO_READS_FAST)				fbcon_bmove(vc, t + count, 0, t, 0,					    b - t - count, vc->vc_cols);			else				goto redraw_up;			fbcon_clear(vc, b - count, 0, count, vc->vc_cols);			break;		case SCROLL_PAN_REDRAW:			if ((p->yscroll + count <=			     2 * (p->vrows - vc->vc_rows))			    && ((!scroll_partial && (b - t == vc->vc_rows))				|| (scroll_partial				    && (b - t - count >					3 * vc->vc_rows >> 2)))) {				if (t > 0)					fbcon_redraw_move(vc, p, 0, t, count);				ypan_up_redraw(vc, t, count);				if (vc->vc_rows - b > 0)					fbcon_redraw_move(vc, p, b - count,							  vc->vc_rows - b, b);			} else				fbcon_redraw_move(vc, p, t + count, b - t - count, t);			fbcon_clear(vc, b - count, 0, count, vc->vc_cols);			break;		case SCROLL_PAN_MOVE:			if ((p->yscroll + count <=			     2 * (p->vrows - vc->vc_rows))			    && ((!scroll_partial && (b - t == vc->vc_rows))				|| (scroll_partial				    && (b - t - count >					3 * vc->vc_rows >> 2)))) {				if (t > 0)					fbcon_bmove(vc, 0, 0, count, 0, t,						    vc->vc_cols);				ypan_up(vc, count);				if (vc->vc_rows - b > 0)					fbcon_bmove(vc, b - count, 0, b, 0,						    vc->vc_rows - b,						    vc->vc_cols);			} else if (info->flags & FBINFO_READS_FAST)				fbcon_bmove(vc, t + count, 0, t, 0,					    b - t - count, vc->vc_cols);			else				goto redraw_up;			fbcon_clear(vc, b - count, 0, count, vc->vc_cols);			break;		case SCROLL_REDRAW:		      redraw_up:			fbcon_redraw(vc, p, t, b - t - count,				     count * vc->vc_cols);			accel_clear(vc, info, real_y(p, b - count), 0,					 count, vc->vc_cols);			scr_memsetw((unsigned short *) (vc->vc_origin +							vc->vc_size_row *							(b - count)),				    vc->vc_video_erase_char,				    vc->vc_size_row * count);			return 1;		}		break;	case SM_DOWN:		if (count > vc->vc_rows)	/* Maximum realistic size */			count = vc->vc_rows;		switch (p->scrollmode) {		case SCROLL_MOVE:			accel_bmove(vc, info, t, 0, t + count, 0,					 b - t - count, vc->vc_cols);			accel_clear(vc, info, t, 0, count, vc->vc_cols);			break;		case SCROLL_WRAP_MOVE:			if (b - t - count > 3 * vc->vc_rows >> 2) {				if (vc->vc_rows - b > 0)					fbcon_bmove(vc, b, 0, b - count, 0,						    vc->vc_rows - b,						    vc->vc_cols);				ywrap_down(vc, count);				if (t > 0)					fbcon_bmove(vc, count, 0, 0, 0, t,						    vc->vc_cols);			} else if (info->flags & FBINFO_READS_FAST)				fbcon_bmove(vc, t, 0, t + count, 0,					    b - t - count, vc->vc_cols);			else				goto redraw_down;			fbcon_clear(vc, t, 0, count, vc->vc_cols);			break;		case SCROLL_PAN_MOVE:			if ((count - p->yscroll <= p->vrows - vc->vc_rows)			    && ((!scroll_partial && (b - t == vc->vc_rows))				|| (scroll_partial				    && (b - t - count >					3 * vc->vc_rows >> 2)))) {				if (vc->vc_rows - b > 0)					fbcon_bmove(vc, b, 0, b - count, 0,						    vc->vc_rows - b,						    vc->vc_cols);				ypan_down(vc, count);				if (t > 0)					fbcon_bmove(vc, count, 0, 0, 0, t,						    vc->vc_cols);			} else if (info->flags & FBINFO_READS_FAST)				fbcon_bmove(vc, t, 0, t + count, 0,					    b - t - count, vc->vc_cols);			else				goto redraw_down;			fbcon_clear(vc, t, 0, count, vc->vc_cols);			break;		case SCROLL_PAN_REDRAW:			if ((count - p->yscroll <= p->vrows - vc->vc_rows)			    && ((!scroll_partial && (b - t == vc->vc_rows))				|| (scroll_partial				    && (b - t - count >					3 * vc->vc_rows >> 2)))) {				if (vc->vc_rows - b > 0)					fbcon_redraw_move(vc, p, b, vc->vc_rows - b,							  b - count);				ypan_down_redraw(vc, t, count);				if (t > 0)					fbcon_redraw_move(vc, p, count, t, 0);			} else				fbcon_redraw_move(vc, p, t, b - t - count, t + count);			fbcon_clear(vc, t, 0, count, vc->vc_cols);			break;		case SCROLL_REDRAW:		      redraw_down:			fbcon_redraw(vc, p, b - 1, b - t - count,				     -count * vc->vc_cols);			accel_clear(vc, info, real_y(p, t), 0, count,					 vc->vc_cols);			scr_memsetw((unsigned short *) (vc->vc_origin +							vc->vc_size_row *							t),				    vc->vc_video_erase_char,				    vc->vc_size_row * count);			return 1;		}	}	return 0;}static void fbcon_bmove(struct vc_data *vc, int sy, int sx, int dy, int dx,			int height, int width){	struct fb_info *info = registered_fb[(int) con2fb_map[vc->vc_num]];	struct display *p = &fb_display[vc->vc_num];		if (!info->fbops->fb_blank && console_blanked)		return;	if (!width || !height)		return;	/*  Split blits that cross physical y_wrap case.	 *  Pathological case involves 4 blits, better to use recursive	 *  code rather than unrolled case	 *	 *  Recursive invocations don't need to erase the cursor over and	 *  over again, so we use fbcon_bmove_rec()	 */	fbcon_bmove_rec(vc, p, sy, sx, dy, dx, height, width,			p->vrows - p->yscroll);}static void fbcon_bmove_rec(struct vc_data *vc, struct display *p, int sy, int sx, 			    int dy, int dx, int height, int width, u_int y_break){	struct fb_info *info = registered_fb[(int) con2fb_map[vc->vc_num]];	u_int b;	if (sy < y_break && sy + height > y_break) {		b = y_break - sy;		if (dy < sy) {	/* Avoid trashing self */			fbcon_bmove_rec(vc, p, sy, sx, dy, dx, b, width,					y_break);			fbcon_bmove_rec(vc, p, sy + b, sx, dy + b, dx,					height - b, width, y_break);		} else {			fbcon_bmove_rec(vc, p, sy + b, sx, dy + b, dx,					height - b, width, y_break);			fbcon_bmove_rec(vc, p, sy, sx, dy, dx, b, width,					y_break);		}		return;	}	if (dy < y_break && dy + height > y_break) {		b = y_break - dy;		if (dy < sy) {	/* Avoid trashing self */			fbcon_bmove_rec(vc, p, sy, sx, dy, dx, b, width,					y_break);			fbcon_bmove_rec(vc, p, sy + b, sx, dy + b, dx,					height - b, width, y_break);		} else {			fbcon_bmove_rec(vc, p, sy + b, sx, dy + b, dx,					height - b, width, y_break);			fbcon_bmove_rec(vc, p, sy, sx, dy, dx, b, width,					y_break);		}		return;	}	accel_bmove(vc, info, real_y(p, sy), sx, real_y(p, dy), dx,			height, width);}static __inline__ void updatescrollmode(struct display *p, struct fb_info *info,					struct vc_data *vc){	int fh = vc->vc_font.height;	int cap = info->flags;	int good_pan = (cap & FBINFO_HWACCEL_YPAN)		 && divides(info->fix.ypanstep, vc->vc_font.height)		 && info->var.yres_virtual > info->var.yres;	int good_wrap = (cap & FBINFO_HWACCEL_YWRAP)		 && divides(info->fix.ywrapstep, vc->vc_font.height)		 && divides(vc->vc_font.height, info->var.yres_virtual);	int reading_fast = cap & FBINFO_READS_FAST;	int fast_copyarea = (cap & FBINFO_HWACCEL_COPYAREA) && !(cap & FBINFO_HWACCEL_DISABLED);	int fast_imageblit = (cap & FBINFO_HWACCEL_IMAGEBLIT) && !(cap & FBINFO_HWACCEL_DISABLED);	p->vrows = info->var.yres_virtual/fh;	if (info->var.yres > (fh * (vc->vc_rows + 1)))		p->vrows -= (info->var.yres - (fh * vc->vc_rows)) / fh;	if ((info->var.yres % fh) && (info->var.yres_virtual % fh <				      info->var.yres % fh))		p->vrows--;	if (good_wrap || good_pan) {		if (reading_fast || fast_copyarea)			p->scrollmode = good_wrap ? SCROLL_WRAP_MOVE : SCROLL_PAN_MOVE;		else			p->scrollmode = good_wrap ? SCROLL_REDRAW :				SCROLL_PAN_REDRAW;	} else {		if (reading_fast || (fast_copyarea && !fast_imageblit))			p->scrollmode = SCROLL_MOVE;		else			p->scrollmode = SCROLL_REDRAW;	}}static int fbcon_resize(struct vc_data *vc, unsigned int width, 			unsigned int height){	struct fb_info *info = registered_fb[(int) con2fb_map[vc->vc_num]];	struct display *p = &fb_display[vc->vc_num];	struct fb_var_screeninfo var = info->var;	int err; int x_diff, y_diff;	int fw = vc->vc_font.width;	int fh = vc->vc_font.height;	var.xres = width * fw;	var.yres = height * fh;	x_diff = info->var.xres - var.xres;	y_diff = info->var.yres - var.yres;	if (x_diff < 0 || x_diff > fw || (y_diff < 0 || y_diff > fh)) {		char mode[40];		DPRINTK("attempting resize %ix%i\n", var.xres, var.yres);		snprintf(mode, 40, "%ix%i", var.xres, var.yres);		err = fb_find_mode(&var, info, mode, info->monspecs.modedb,				   info->monspecs.modedb_len, NULL,				   info->var.bits_per_pixel);		if (!err || width > var.xres/fw || height > var.yres/fh)			return -EINVAL;		DPRINTK("resize now %ix%i\n", var.xres, var.yres);		if (CON_IS_VISIBLE(vc)) {			var.activate = FB_ACTIVATE_NOW |				FB_ACTIVATE_FORCE;			fb_set_var(info, &var);			info->flags &= ~FBINFO_MISC_MODESWITCH;		}	}	updatescrollmode(p, info, vc);	return 0;}static int fbcon_switch(struct vc_data *vc){	struct fb_info *info = registered_fb[(int) con2fb_map[vc->vc_num]];	struct display *p = &fb_display[vc->vc_num];	int i;	if (softback_top) {		int l = fbcon_softback_size / vc->vc_size_row;		if (softback_lines)			fbcon_set_origin(vc);		softback_top = softback_curr = softback_in = softback_buf;		softback_lines = 0;		if (l > 5)			softback_end = softback_buf + l * vc->vc_size_row;		else {			/* Smaller scrollback makes no sense, and 0 would screw			   the operation totally */			softback_top = 0;		}	}	if (logo_shown >= 0) {		struct vc_data *conp2 = vc_cons[logo_shown].d;		if (conp2->vc_top == logo_lines		    && conp2->vc_bottom == conp2->vc_rows)			conp2->vc_top = 0;		logo_shown = -1;	}	if (info)		info->var.yoffset = p->yscroll = 0;	/*	 * FIXME: If we have multiple fbdev's loaded, we need to	 * update all info->currcon.  Perhaps, we can place this	 * in a centralized structure, but this might break some	 * drivers.	 *	 * info->currcon = vc->vc_num;	 */	for (i = 0; i < FB_MAX; i++) {		if (registered_fb[i] != NULL)			registered_fb[i]->currcon = vc->vc_num;	} 	fbcon_resize(vc, vc->vc_cols, vc->vc_rows);	if (info->flags & FBINFO_MISC_MODESWITCH) {		if (info->fbops->fb_set_par)			info->fbops->fb_set_par(info);		info->flags &= ~FBINFO_MISC_MODESWITCH;	}	switch (p->scrollmode) {	case SCROLL_WRAP_MOVE:		scrollback_phys_max = p->vrows - vc->vc_rows;		break;	case SCROLL_PAN_MOVE:	case SCROLL_PAN_REDRAW:		scrollback_phys_max = p->vrows - 2 * vc->vc_rows;		if (scrollback_phys_max < 0)			scrollback_phys_max = 0;		break;	default:		scrollback_phys_max = 0;		break;	}	scrollback_max = 0;	scrollback_current = 0;	update_var(vc->vc_num, info);	fbcon_set_palette(vc, color_table); 		if (vt_cons[vc->vc_num]->vc_mode == KD_TEXT)		accel_clear_margins(vc, info, 0);	if (logo_shown == -2) {		logo_shown = fg_console;		/* This is protected above by initmem_freed */		fb_show_logo(info);		update_region(fg_console,			      vc->vc_origin + vc->vc_size_row * vc->vc_top,			      vc->vc_size_row * (vc->vc_bottom -						 vc->vc_top) / 2);		return 0;	}	return 1;}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?