- October 29th, 2009
- 9:50 pm
Ubuntu 9.10 (Karmic Koala) is out, so I decided to upgrade. The upgrade process itself went fine, but it seems there were some other changes to the system, which affected me.
I'm running on a MacBook Pro Santa Rosa with 4 GB mem, which means that if I use the standard 32 bit kernel, I cannot use all of the memory. That's not really a problem, I just recompile the kernel to support PAE.
However, I also use XMonad (because it's the best window manager available), and previously I launched it from ~/.xsession. However, the new version of GDM doesn't support that anymore, so now you have to drop a file in /usr/share/xsessions/xmonad.desktop - mine contains this:
CODE:
-
[Desktop Entry]
-
Encoding=UTF-8
-
Name=xmonad
-
Comment=xmonad
-
Exec=/pack/haskell/bin/runxmonad.sh
-
Type=Application
Other than that, simply follow this little guide to make the keyboard and touchpad work properly.
I've just finished upgrading my two machines (a MacBook Pro, Santa Rosa, and an iMac) to Ubuntu 9.04 (Jaunty). Everything went pretty smoothless, although the download did take some time.
I have only made two adjustments:
- Fn keys enabled by default - previously, you would add "options hid pb_fnmode=2" to a file in /etc/modprobe.d - now it should be "options hid_apple fnmode=2" instead. This can also be changed at runtime in /sys/module/hid_apple/parameters/fnmode
- Compiled a new kernel with highmem support (by following this little guide)
Other than that, everything seems to work without any trouble, and nothing broke. Sweet.
- December 29th, 2007
- 3:02 pm
The coming of a new year usually attracts a number of predictions, some more accurate than others. I'm no good at predicting, so I'll give my wishes for the new year instead:
- Get my Macbook Pro to work with the DVI-VGA adapter (done)
- I'd like to be convinced that the dynamic languages are really the way forward. I see that they can do smart stuff, but I sorely miss some kind of help using whatever APIs are developed, besides the source code, and I can't get used to working without a proper editor. I can probably live without the editor, but having methods like def initialize (options = {}) just hurts my eyes. I haven't used Ruby/Rails that much, but I've already dug around more source code than I care to in order to find out what the legal options were.
- Release Java 7 with proper closures. This probably won't happen, and even when it does, it will take a couple of years before it's adopted.
- Already, I know that I'm going to spend a good part of my time fiddling around with huge WSDLs and schemas. Could somebody find a way to stop the black hole that is Web Services?
- As for Linux on the desktop, that will still not happen, even though it should, especially when you look at the new distributions such as Ubuntu. However, I'm pretty satisfied, now I just need xmonad to support dynamic twinview, so I can add a monitor without restarting X
- And to Apple: please take corporate customers seriously and make a docking station for the Macbook (Pro). I'm getting tired of plugging and unplugging 5 different cables every time I have to move my laptop.
- Less focus on creating big monolithic systems, more on distributing across autonomous nodes.
- Finally, it would be great with a proper vi-mode for Eclipse - like vimperator for Firefox.
- December 29th, 2007
- 2:36 am
I recently discovered that using the DVI adapter with the Macbook Pro wasn't as easy as I thought. The MBP comes with a DVI connector, which is great, and it works just fine when you plug a DVI monitor into it. However, when doing presentations, you almost always have to use a regular VGA connector. Even though the driver (nvidia) could see the screen, it refused to display anything.
Now I've finally worked out a solution, and luckily it's not that complicated. The trick is that even though the monitor is recognized, it is still treated as a DVI monitor. In order to get the driver to treat it as a normal VGA monitor, add this to xorg.conf:
CODE:
-
Section "Device"
-
...
-
Option "ConnectedMonitor" "DFP-0,DFP-1,CRT-0"
-
Option "MetaModes" "DFP-0: 1440x900, DFP-1: 1680x1050; DFP-0: 1440x900, CRT-0: 1280x1024"
-
EndSection
Of course, the modes should be adjusted according to the setup. The MetaModes line should contain the valid combinations, separated by ;. In this case, I've specified that the internal display should always run at 1440x900, and then I either have a DVI monitor connected at 1680x1050 or a VGA monitor at 1280x1024. Of course, it's also possible just to specify one monitor if you're not using TwinView.
The trick now is that when running in X, xrandr can be used to switch between the monitors. When running xrandr, it should display something like this:
CODE:
-
Screen 0: minimum 1280 x 900, current 1440 x 900, maximum 2720 x 1024
-
default connected 1440x900+0+0 (normal left inverted right) 0mm x 0mm
-
1440x900 50.0* 51.0 54.0
-
2720x1024 52.0 50.0
-
2464x900 53.0
-
1280x1024 55.0
The tricky part now is that the nvidia driver "hides" the monitor configuration in the refresh rate. For example, the 1280x1024 resolution has only one rate, which means that this resolution is unique to one monitor. Switching to this mode with 'xrandr -s 1280x1024' will enable the VGA monitor and disable the internal display.
However, the mode 2720x1024 has two refresh rates, 52 and 50. Unfortunately there's no easy way of telling it, but the 52 rate is with the internal display and the VGA monitor on, while the 50 is with only the internal display on (and a larger desktop, something you probably don't want to use). Switching on the VGA monitor then requires two arguments: xrandr -s 2720x1024 --rate 52.
This isn't the easiest to work with, so I've created a small Perl script which can query the nvidia driver directly to find the available modes, and then call xrandr. Download the script here. To use it, the X11::Protocol Perl package must be installed, and of course xrandr and the nvidia driver. To use the script, simply call it without any arguments to get a list of the supported resolutions and monitors. Select a configuration by calling the script with -s where id is one of the numbers in the first column. That's about as simple as it gets.
- December 16th, 2007
- 12:57 am
xmonad 0.5 is out, and I haven't upgraded my installation in a while. It's pretty nice that configuration is now done in /.xmonad/xmonad.hs, then there's no need to recompile xmonad all the time. The configuration change did require a reorganisation of my old Config.hs, so I took a look at some of the samples to see what other people did.
I ended up with something which looks like this:
This is my xmonad.hs:
CODE:
-
-- XMonad Core
-
import XMonad
-
import XMonad.Layout
-
import XMonad.Operations
-
import qualified XMonad.StackSet as W
-
-
-- GHC hierarchical libraries
-
import Data.Bits ((.|.))
-
import qualified Data.Map as M
-
import Graphics.X11
-
import Graphics.X11.Xlib
-
import Graphics.X11.Xlib.Extras
-
import System.IO
-
-
-- Contribs
-
import XMonad.Actions.CycleWS
-
import XMonad.Actions.SwapWorkspaces
-
import XMonad.Actions.Submap
-
import XMonad.Actions.WindowBringer
-
import XMonad.Actions.FloatKeys
-
-
import XMonad.Hooks.UrgencyHook
-
import XMonad.Layout.NoBorders
-
import XMonad.Layout.Tabbed
-
import XMonad.Layout.WindowNavigation
-
import XMonad.Layout.Grid
-
import XMonad.Layout.LayoutHints
-
import XMonad.Layout.Dishes
-
import XMonad.Util.EZConfig
-
import XMonad.Util.Run
-
-
import XMonad.Prompt.Shell
-
import XMonad.Prompt
-
-
import XMonad.Hooks.DynamicLog ( PP(..), dynamicLogWithPP, dzenColor, wrap, defaultPP )
-
-
myfont = "\"-xos4-terminus-medium-r-normal--12-120-72-72-c-60-iso8859-1\""
-
fgcolor = "black"
-
bgcolor = "white"
-
-
statusBarCmd= "dzen2 -e '' -w 660 -h 15 -ta l -xs 1 -fg " ++ fgcolor ++ " -bg " ++ bgcolor ++ " -fn " ++ myfont
-
-
-- Get ready!
-
main = do din <- spawnPipe statusBarCmd
-
xmonad $ withUrgencyHook dzenUrgencyHook { args = ["-bg", "darkgreen", "-xs", "1"] }
-
$ defaultConfig
-
{ workspaces = workspaces'
-
, modMask = modMask'
-
, numlockMask = 0
-
, layoutHook = layoutHook'
-
, terminal = "urxvtc || urxvt"
-
, normalBorderColor = "#dddddd"
-
, focusedBorderColor = "#3499dd"
-
, defaultGaps = [(15,0,0,0)]
-
, logHook = dynamicLogWithPP $ myPP din
-
}
-
`additionalKeys` keys'
-
-
modMask' = mod4Mask
-
-
workspaces' = map show [1] ++ ["web", "mail", "chat", "code"] ++ map show [6 .. 9 :: Int]
-
-
layoutHook' =
-
configurableNavigation noNavigateBorders $
-
layouts
-
-
layouts =
-
Mirror tiled
-
||| tiled
-
||| Grid
-
||| layoutHints Full
-
||| Dishes 2 (1/5)
-
||| noBorders (tabbed shrinkText
-
defaultTConf { fontName = myfont })
-
where
-
tiled = Tall nmaster delta ratio
-
nmaster = 2 -- The default number of windows in the master pane
-
ratio = 1/2 -- Default proportion of screen occupied by master pane
-
delta = 3/100 -- Percent of screen to increment by when resizing panes
-
-
noFollow CrossingEvent {} = return False
-
noFollow _ = return True
-
-
keys' =
-
[ ((modMask' .|. shiftMask, xK_d ), spawn "date | dzen2 -p 2 -xs 1") -- %! Print current date
-
, ((modMask', xK_p), shellPrompt defaultXPConfig)
-
]
-
++
-
-- modMask'-[1..0] %! Switch to workspace N
-
-- modMask'-shift-[1..0] %! Move client to workspace N
-
[((m .|. modMask', k), windows $ f i)
-
| (i, k) <- zip workspaces' $ [xK_1 .. xK_9] ++ [xK_0]
-
, (f, m) <- [(W.greedyView, 0), (W.shift, shiftMask)]]
-
++
-
-- modMask'-{e,r} %! Switch to physical/Xinerama screens 1 or 2
-
-- modMask'-shift-{e,r} %! Move client to screen 1 or 2
-
[((m .|. modMask', key), screenWorkspace sc>>= flip whenJust (windows . f))
-
| (key, sc) <- zip [xK_e, xK_w] [0..]
-
, (f, m) <- [(W.view, 0), (W.shift, shiftMask)]]
-
++
-
[((modMask' .|. mod1Mask, k), windows $ swapWithCurrent i)
-
| (i, k) <- zip workspaces' $ [xK_1 .. xK_9] ++ [xK_0]]
-
-- float keys
-
++
-
[
-
((modMask', xK_d ), withFocused (keysResizeWindow (-10,-10) (1,1)))
-
, ((modMask', xK_s ), withFocused (keysResizeWindow (10,10) (1,1)))
-
, ((modMask', xK_a ), withFocused (keysMoveWindowTo (512,384) (1, 0)))
-
]
-
-
-
myPP h = defaultPP
-
{ ppCurrent = dzenColor "white" "#cd8b00" . pad
-
, ppVisible = dzenColor "white" "#666666" . pad
-
, ppHidden = dzenColor "black" "#cccccc" . pad
-
, ppHiddenNoWindows = dzenColor "#999999" "#cccccc" . pad
-
, ppWsSep = dzenColor "#bbbbbb" "#cccccc" "^r(1x18)"
-
, ppSep = dzenColor "#bbbbbb" "#cccccc" "^r(1x18)"
-
, ppLayout = dzenColor "black" "#cccccc" .
-
(\ x -> case x of
-
"TilePrime Horizontal" ->
-
" ^i(/home/emertens/images/tile_horz.xpm) "
-
"TilePrime Vertical" ->
-
" ^i(/home/emertens/images/tile_vert.xpm) "
-
"Hinted Full" ->
-
" ^i(/home/emertens/images/fullscreen.xpm) "
-
_ -> pad x
-
)
-
, ppTitle = (' ':) . escape
-
, ppOutput = hPutStrLn h
-
}
-
where
-
escape = concatMap (\x -> if x == '^' then "^^" else [x])
-
pad = wrap " " " "
I use a simple shell script for starting xmonad:
CODE:
-
ssh-add ~/.ssh/id_dsa </dev/null>/dev/null
-
Esetroot /usr/share/backgrounds/warty-final-ubuntu.png
-
unclutter -idle 2 &
-
urxvtd -f -o
-
irxevent -d
-
-
gnome-settings-daemon &
-
gnome-volume-manager &
-
-
export PATH=$PATH:`dirname $0`
-
`dirname $0`/status &
-
-
xmonad
Finally, the dzen2 status bar is run using this script:
CODE:
-
#!/bin/sh
-
-
BG=white
-
FG=black
-
FONT="-xos4-terminus-medium-r-normal--12-120-72-72-c-60-iso8859-1"
-
-
BFG="#444"
-
-
gcpubar -i 2 -fg '#444' -w 30 -h 10 | dzen2 -e '' -x 1360 -fg $FG \
-
-bg $BG -fn $FONT -h 15 -xs 1 &
-
-
while :; do
-
MEM=`memstatus.awk /proc/meminfo`
-
CPU=`cpustatus.awk`
-
DATE=`date +"%d-%m %k:%M"`
-
-
REM=`awk '/remaining capacity/ { print $3 }' /proc/acpi/battery/BAT0/state`
-
LAST=`awk '/last full/ { print $4}' /proc/acpi/battery/BAT0/info`
-
STATE=`awk '{print $2}' /proc/acpi/ac_adapter/ADP1/state`
-
if [ "$STATE" = "on-line" ]; then
-
BAT=$(echo $REM $LAST | awk '{printf "Bat: %.1f%%, AC", ($1/$2)*100'})
-
else
-
PRESENT=`awk '/present rate/ { print $3}' /proc/acpi/battery/BAT0/state`
-
BAT=$(echo $REM $LAST $PRESENT | \
-
awk '{printf "Bat: %.1f%%, %d min", ($1/$2)*100, ($1/$3)*60}')
-
fi
-
-
LOAD=`awk '{print $1 " " $2 " " $3}' /proc/loadavg`
-
-
echo "$CPU $MEM | $DATE | $BAT | $LOAD"
-
sleep 5
-
done | dzen2 -e '' -x 660 -w 700 -fg $FG -bg $BG -fn $FONT -h 15 -xs 1
Two awk scripts are used - one for the memory usage and one for CPU frequencies. Why awk? No idea, it just seemed like a good choice at the time.
memstatus.awk:
CODE:
-
#!/usr/bin/awk -f
-
BEGIN {
-
BG="darkgrey";
-
FG="#444";
-
WIDTH=30;
-
HEIGHT=10;
-
}
-
-
/MemTotal/ {t=$2};
-
/MemFree/ {f=$2};
-
/SwapTotal/ {st=$2};
-
/SwapFree/ {sf=$2};
-
-
END {
-
mw=int(WIDTH*(t-f)/t);
-
fw=WIDTH-mw;
-
sw=int(WIDTH*(st-sf)/st);
-
-
printf("Mem: ^ib(1)^fg(%s)^r(%dx%d)^fg(%s)^r(%dx%d)" \
-
"^fg(%s)^r(2x%d)^fg(%s)^r(%dx%d)^fg(%s)^r(%dx%d)^ib(0)^fg()\n",
-
FG, mw, HEIGHT,
-
BG, fw, HEIGHT,
-
"black",
-
HEIGHT, BG, WIDTH-sw,
-
HEIGHT, FG, sw, HEIGHT);
-
}
CODE:
-
#!/usr/bin/awk -f
-
BEGIN {
-
BG="darkgrey"
-
FG="#444"
-
WIDTH=30
-
HEIGHT=10
-
CPUS=0
-
-
cmd = "ls /sys/devices/system/cpu"
-
while ((cmd | getline)> 0) {
-
if ($1 ~ /cpu/) {
-
CPUS++
-
}
-
}
-
close(cmd)
-
-
printf "CPU: "
-
for (i=0; i <CPUS; i++) {
-
cfn="/sys/devices/system/cpu/cpu" i "/cpufreq/scaling_cur_freq"
-
mfn="/sys/devices/system/cpu/cpu" i "/cpufreq/scaling_max_freq"
-
getline cf <cfn
-
getline mf <mfn
-
cw=int(WIDTH*(mf-cf)/mf)
-
-
printf("^ib(1)^fg(%s)^r(%dx%d)^fg(%s)^r(%dx%d)^ib(0)^fg()",
-
FG, cw, HEIGHT, BG, WIDTH-cw, HEIGHT)
-
-
close(cfn)
-
close(mfn)
-
-
if (i <CPUS - 1) {
-
printf("^ib(0)^fg()^r(5x0)^ib(1)")
-
}
-
-
}
-
printf("\n");
-
}
- October 28th, 2007
- 5:44 pm
I managed to get the remote control to work with my MacBook Pro and Ubuntu. It should work out of the box according to almost everybody when you google for it, but it just didn't. I know, I said ealier that I didn't know what to use it for, but that doesn't really matter. What matters is that I had a piece of hardware which didn't work, and that's just not good enough.
After digging around a little in lsusb and the module source code, I found out that on the most recent MacBooks, the IR receiver has been assigned a new device id. The original receivers had id 0x8240 while mine has 0x8242. It doesn't look like anything else has changed, and after recompiling the module with the correct device it, it suddenly worked like it should.
Compiling the module isn't exactly for the faint of heart, so I won't describe the details here - let me know if you want it, and I'll post it. However, it does include "apt-get source linux-ubuntu-modules-2.6.22-14-generic".
Also, remember that the appleir module must be loaded before usbhid. To do this on a running system do
"rmmod appleir usbhid; modprobe appleir; modprobe udbhid"
Remember to do it on one line, or you will lose keyboard input after running rmmod.
Other than this, follow this guide to configure the remote control. So far, my .lircrc contains the following, supporting slide change in OOImpress and Amarok:
CODE:
-
begin
-
prog = irxevent
-
button = KEY_MENU
-
button = KEY_PP
-
config = Key space amarokapp
-
flags = quit
-
end
-
begin
-
prog = irxevent
-
button = KEY_MENU
-
button = KEY_N
-
config = Key ctrl-alt-b amarokapp
-
flags = quit
-
end
-
-
begin
-
prog = irxevent
-
button = KEY_MENU
-
button = KEY_MENU
-
config = Key Escape CurrentWindow
-
end
-
-
-
begin
-
prog = irxevent
-
button = KEY_P
-
config = Key Page_Up CurrentWindow
-
end
-
begin
-
prog = irxevent
-
button = KEY_N
-
config = Key Page_Down CurrentWindow
-
end
Btw, I've filed a bug with Ubuntu, so hopefully the module will be fixed at some point.
- October 27th, 2007
- 12:37 am
as I wrote about yesterday, I had some problems getting software suspend to work with Ubuntu and my MacBook Pro. Well, it looks like the problem has been solved. As reported here, it looks like there's a bug in the kernel, so downgrading a little should fix it. However, in my case I also had to disable gnome-power-manager - not really an issue. After changing these two things, I've been able to suspend and resume without problems - at least for now.
- October 25th, 2007
- 10:54 pm
After reinstalling xmonad on my laptop using Ubuntu as the base distribution, I've finally got a setup I like, so I thought I'd share it. So, here is my script which starts xmonad:
CODE:
-
ssh-add ~/.ssh/id_dsa </dev/null>/dev/null
-
-
Esetroot /usr/share/backgrounds/warty-final-ubuntu.png
-
unclutter -idle 2 &
-
-
urxvtd -f -o
-
-
gnome-settings-daemon &
-
gnome-power-manager
-
gnome-volume-manager &
-
-
BG=white
-
FG=black
-
FONT="-xos4-terminus-medium-r-normal--12-120-72-72-c-60-iso8859-1"
-
-
export PATH=$PATH:`dirname $0`
-
-
# create a pipe for xmonad to talk to
-
PIPE=$HOME/.xmonad-status
-
rm -f $PIPE
-
mkfifo -m 600 $PIPE
-
[ -p $PIPE ] || exit
-
-
# and a workspace status bar, reading's xmonad's stdout
-
dzen2 -e '' -w 680 -ta l -fg $FG -bg $BG -fn $FONT -h 15 <$PIPE &
-
-
`dirname $0`/status &
-
-
xmonad> $PIPE &
-
wait $!
-
-
pkill -HUP dzen2
-
-
wait
This script is launched from ~/.xsession. It uses dzen and some of the dzen gadgets (which must be compiled separately from dzen/gadgets). It also starts some of the Gnome daemons which normally run under Ubuntu. This means that power management and sound volume goes through Gnome, and devices are mounted automatically when they are added to the system.
Two dzens are launched: One for displaying workspaces and one for general status. The general status script looks like this:
CODE:
-
#!/bin/sh
-
-
BG=white
-
FG=black
-
FONT="-xos4-terminus-medium-r-normal--12-120-72-72-c-60-iso8859-1"
-
-
while :; do
-
MEM=`awk '/MemTotal/ {t=$2}; /MemFree/ {f=$2}; END {print t-f " " t}' \
-
/proc/meminfo | dbar -s ':' -l 'Mem:' -w 10`
-
CF0=$(echo `cat /sys/devices/system/cpu/cpu0/cpufreq/scaling_cur_freq` \
-
`cat /sys/devices/system/cpu/cpu0/cpufreq/scaling_max_freq` | dbar -w 7 -l 'CPU0:')
-
CF1=$(echo `cat /sys/devices/system/cpu/cpu1/cpufreq/scaling_cur_freq` \
-
`cat /sys/devices/system/cpu/cpu1/cpufreq/scaling_max_freq` | dbar -w 7 -l 'CPU1:')
-
DATE=`date +"%d-%m %k:%M"`
-
-
REM=`grep 'remaining capacity' /proc/acpi/battery/BAT0/state | awk '{print $3}'`
-
LAST=`grep 'last full' /proc/acpi/battery/BAT0/info |awk '{print $4}'`
-
STATE=`awk '{print $2}' /proc/acpi/ac_adapter/ADP1/state`
-
if [ "$STATE" = "on-line" ]; then
-
BAT=$(echo $REM $LAST | awk '{printf "Bat: %.1f%%, AC", ($1/$2)*100'})
-
else
-
PRESENT=`grep 'present rate' /proc/acpi/battery/BAT0/state | awk '{print $3}'`
-
BAT=$(echo $REM $LAST $PRESENT | awk '{printf "Bat: %.1f%%, %d min", ($1/$2)*100, ($1/$3)*60}')
-
fi
-
-
LOAD=`awk '{print $1 " " $2 " " $3}' /proc/loadavg`
-
-
echo "$CF0 $CF1 $MEM | $DATE | $BAT | $LOAD"
-
sleep 5
-
done | dzen2 -e '' -x 440 -w 1000 -fg $FG -bg $BG -fn $FONT -h 15
The script is made for a MacBook Pro with a two core processor and acpi. It displays the CPU frequencies in a dbar, memory usage, battery status, and load. I would like to display network status too, but I can't figure out how to dig around in dbus efficiently, so that's still missing.