diff options
author | Michael Olson <mwolson@gnu.org> | 2006-08-27 05:21:00 +0000 |
---|---|---|
committer | Michael Olson <mwolson@gnu.org> | 2006-08-27 05:21:00 +0000 |
commit | fa598825e414642e9092f93cde236052c7141827 (patch) | |
tree | 6721a6bc4e38c7da22e2e86a4d71e2251659f761 | |
parent | 2f052eb2aae8ff9ee72129cda8d3bc84fb80c354 (diff) |
Include the tq.el file from Emacs 22 with EMMS, instead of re-implementing it in emms-player-mpd.el.
darcs-hash:20060827052155-1bfb2-b2b47f20d03eaa85cecdf3c1805279edc59c68ee.gz
-rw-r--r-- | emms-player-mpd.el | 119 | ||||
-rw-r--r-- | tq.el | 171 |
2 files changed, 180 insertions, 110 deletions
diff --git a/emms-player-mpd.el b/emms-player-mpd.el index a748c62..603a0f8 100644 --- a/emms-player-mpd.el +++ b/emms-player-mpd.el @@ -102,6 +102,7 @@ (require 'emms-player-simple) (require 'emms-source-playlist) ; for emms-source-file-parse-playlist +(require 'tq) (defgroup emms-player-mpd nil "EMMS player for MusicPD." @@ -224,107 +225,6 @@ If your EMMS playlist contains stored playlists, set this to nil." 'seek 'emms-player-mpd-seek) -;;; Transaction Queue (with some improvements) based on tq.el - -(defun emms-player-mpd-tq-create (process) - "Create and return a transaction queue communicating with PROCESS. -PROCESS should be a subprocess capable of sending and receiving -streams of bytes. It may be a local process, or it may be connected -to a tcp server on another machine." - (let ((tq (cons nil (cons process - (generate-new-buffer - (concat " emms-player-mpd-tq-temp-" - (process-name process))))))) - (set-process-filter process - `(lambda (proc string) - (emms-player-mpd-tq-filter ',tq string))) - tq)) - -;; accessors -(defun emms-player-mpd-tq-queue (tq) - (car tq)) -(defun emms-player-mpd-tq-process (tq) - (car (cdr tq))) -(defun emms-player-mpd-tq-buffer (tq) - (cdr (cdr tq))) -(defun emms-player-mpd-tq-queue-head-question (tq) - (car (car (emms-player-mpd-tq-queue tq)))) -(defun emms-player-mpd-tq-queue-head-regexp (tq) - (car (cdr (car (emms-player-mpd-tq-queue tq))))) -(defun emms-player-mpd-tq-queue-head-closure (tq) - (car (cdr (cdr (car (emms-player-mpd-tq-queue tq)))))) -(defun emms-player-mpd-tq-queue-head-fn (tq) - (cdr (cdr (cdr (car (emms-player-mpd-tq-queue tq)))))) - -(defun emms-player-mpd-tq-queue-empty (tq) - (not (emms-player-mpd-tq-queue tq))) - -(defun emms-player-mpd-tq-queue-add (tq question re closure fn) - (setcar tq (nconc (emms-player-mpd-tq-queue tq) - (cons (cons question (cons re (cons closure fn))) nil))) - 'ok) - -(defun emms-player-mpd-tq-queue-pop (tq) - (setcar tq (cdr (car tq))) - (let ((question (emms-player-mpd-tq-queue-head-question tq))) - (when question - (process-send-string (emms-player-mpd-tq-process tq) question))) - (null (car tq))) - -(defun emms-player-mpd-tq-enqueue (tq question regexp closure fn) - "Add a transaction to transaction queue TQ. -This sends the string QUESTION to the process that TQ communicates with. -When the corresponding answer comes back, we call FN -with two arguments: CLOSURE, and the answer to the question. -REGEXP is a regular expression to match the entire answer; -that's how we tell where the answer ends." - (let ((sendp (not (emms-player-mpd-tq-queue-head-question tq)))) - (emms-player-mpd-tq-queue-add tq question regexp closure fn) - (when sendp - (process-send-string (emms-player-mpd-tq-process tq) question)))) - -(defun emms-player-mpd-tq-close (tq) - "Shut down transaction queue TQ, terminating the process." - (delete-process (emms-player-mpd-tq-process tq)) - (kill-buffer (emms-player-mpd-tq-buffer tq))) - -(defun emms-player-mpd-tq-filter (tq string) - "Append STRING to the TQ's buffer; then process the new data." - (let ((buffer (emms-player-mpd-tq-buffer tq))) - (when (buffer-live-p buffer) - (with-current-buffer buffer - (goto-char (point-max)) - (insert string) - (emms-player-mpd-tq-process-buffer tq))))) - -(defun emms-player-mpd-tq-process-buffer (tq) - "Check TQ's buffer for the regexp at the head of the queue." - (let ((buffer (emms-player-mpd-tq-buffer tq))) - (when (buffer-live-p buffer) - (set-buffer buffer) - (if (= 0 (buffer-size)) () - (if (emms-player-mpd-tq-queue-empty tq) - (let ((buf (generate-new-buffer "*spurious*"))) - (copy-to-buffer buf (point-min) (point-max)) - (delete-region (point-min) (point)) - (pop-to-buffer buf nil) - (error "Spurious communication from process %s, see buffer %s" - (process-name (emms-player-mpd-tq-process tq)) - (buffer-name buf))) - (goto-char (point-min)) - (if (re-search-forward (emms-player-mpd-tq-queue-head-regexp tq) - nil t) - (let ((answer (buffer-substring (point-min) (point)))) - (delete-region (point-min) (point)) - (unwind-protect - (condition-case nil - (funcall (emms-player-mpd-tq-queue-head-fn tq) - (emms-player-mpd-tq-queue-head-closure tq) - answer) - (error nil)) - (emms-player-mpd-tq-queue-pop tq)) - (emms-player-mpd-tq-process-buffer tq)))))))) - ;;; Dealing with the MusicPD network process (defvar emms-player-mpd-process nil) @@ -375,16 +275,15 @@ return at the end of a request.") (set-process-sentinel emms-player-mpd-process 'emms-player-mpd-sentinel) (setq emms-player-mpd-queue - (emms-player-mpd-tq-create emms-player-mpd-process)) + (tq-create emms-player-mpd-process)) (if (fboundp 'set-process-query-on-exit-flag) (set-process-query-on-exit-flag emms-player-mpd-process nil) (process-kill-without-query emms-player-mpd-process)) ;; send password (when (stringp emms-player-mpd-server-password) - (emms-player-mpd-tq-enqueue - emms-player-mpd-queue - (concat "password " emms-player-mpd-server-password "\n") - emms-player-mpd-status-regexp nil #'ignore)))) + (tq-enqueue emms-player-mpd-queue + (concat "password " emms-player-mpd-server-password "\n") + emms-player-mpd-status-regexp nil #'ignore t)))) (defun emms-player-mpd-close-process (&optional from-sentinel) "Terminate the current MusicPD client process. @@ -393,7 +292,7 @@ in which case certain checks should not be made." (when (or from-sentinel (and (processp emms-player-mpd-process) (memq (process-status emms-player-mpd-process) '(run open)))) - (emms-player-mpd-tq-close emms-player-mpd-queue) + (tq-close emms-player-mpd-queue) (setq emms-player-mpd-queue nil) (setq emms-player-mpd-process nil))) @@ -403,9 +302,9 @@ When a reply comes, call FN with CLOSURE and the result." (emms-player-mpd-ensure-process) (unless (string= (substring question -1) "\n") (setq question (concat question "\n"))) - (emms-player-mpd-tq-enqueue emms-player-mpd-queue question - emms-player-mpd-status-regexp - closure fn)) + (tq-enqueue emms-player-mpd-queue question + emms-player-mpd-status-regexp + closure fn t)) ;;; Helper functions @@ -0,0 +1,171 @@ +;;; tq.el --- utility to maintain a transaction queue + +;; Copyright (C) 1985, 1986, 1987, 1992, 2002, 2003, 2004, +;; 2005, 2006 Free Software Foundation, Inc. + +;; Author: Scott Draves <spot@cs.cmu.edu> +;; Maintainer: FSF +;; Adapted-By: ESR +;; Keywords: extensions + +;; This file is part of GNU Emacs. + +;; GNU Emacs is free software; you can redistribute it and/or modify +;; it under the terms of the GNU General Public License as published by +;; the Free Software Foundation; either version 2, or (at your option) +;; any later version. + +;; GNU Emacs 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 General Public License for more details. + +;; You should have received a copy of the GNU General Public License +;; along with GNU Emacs; see the file COPYING. If not, write to the +;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, +;; Boston, MA 02110-1301, USA. + +;;; Commentary: + +;; This file manages receiving a stream asynchronously, parsing it +;; into transactions, and then calling the associated handler function +;; upon the completion of each transaction. + +;; Our basic structure is the queue/process/buffer triple. Each entry +;; of the queue part is a list of question, regexp, closure, and +;; function that is consed to the last element. + +;; A transaction queue may be created by calling `tq-create'. + +;; A request may be added to the queue by calling `tq-enqueue'. If +;; the `delay-question' argument is non-nil, we will wait to send the +;; question to the process until it has finished sending other input. +;; Otherwise, once a request is enqueued, we send the given question +;; immediately to the process. + +;; We then buffer bytes from the process until we see the regexp that +;; was provided in the call to `tq-enqueue'. Then we call the +;; provided function with the closure and the collected bytes. If we +;; have indicated that the question from the next transaction was not +;; sent immediately, send it at this point, awaiting the response. + +;;; Code: + +;;; Accessors + +;; This part looks like (queue . (process . buffer)) +(defun tq-queue (tq) (car tq)) +(defun tq-process (tq) (car (cdr tq))) +(defun tq-buffer (tq) (cdr (cdr tq))) + +;; The structure of `queue' is as follows +;; ((question regexp closure . fn) +;; <other queue entries>) +;; question: string to send to the process +(defun tq-queue-head-question (tq) (car (car (tq-queue tq)))) +;; regexp: regular expression that matches the end of a response from +;; the process +(defun tq-queue-head-regexp (tq) (car (cdr (car (tq-queue tq))))) +;; closure: additional data to pass to the function +(defun tq-queue-head-closure (tq) (car (cdr (cdr (car (tq-queue tq)))))) +;; fn: function to call upon receiving a complete response from the +;; process +(defun tq-queue-head-fn (tq) (cdr (cdr (cdr (car (tq-queue tq)))))) + +;; Determine whether queue is empty +(defun tq-queue-empty (tq) (not (tq-queue tq))) + +;;; Core functionality + +;;;###autoload +(defun tq-create (process) + "Create and return a transaction queue communicating with PROCESS. +PROCESS should be a subprocess capable of sending and receiving +streams of bytes. It may be a local process, or it may be connected +to a tcp server on another machine." + (let ((tq (cons nil (cons process + (generate-new-buffer + (concat " tq-temp-" + (process-name process))))))) + (set-process-filter process + `(lambda (proc string) + (tq-filter ',tq string))) + tq)) + +(defun tq-queue-add (tq question re closure fn) + (setcar tq (nconc (tq-queue tq) + (cons (cons question (cons re (cons closure fn))) nil))) + 'ok) + +(defun tq-queue-pop (tq) + (setcar tq (cdr (car tq))) + (let ((question (tq-queue-head-question tq))) + (when question + (process-send-string (tq-process tq) question))) + (null (car tq))) + +(defun tq-enqueue (tq question regexp closure fn &optional delay-question) + "Add a transaction to transaction queue TQ. +This sends the string QUESTION to the process that TQ communicates with. + +When the corresponding answer comes back, we call FN with two +arguments: CLOSURE, which may contain additional data that FN +needs, and the answer to the question. + +REGEXP is a regular expression to match the entire answer; +that's how we tell where the answer ends. + +If DELAY-QUESTION is non-nil, delay sending this question until +the process has finished replying to any previous questions. +This produces more reliable results with some processes." + (let ((sendp (or (not delay-question) + (not (tq-queue tq))))) + (tq-queue-add tq (unless sendp question) regexp closure fn) + (when sendp + (process-send-string (tq-process tq) question)))) + +(defun tq-close (tq) + "Shut down transaction queue TQ, terminating the process." + (delete-process (tq-process tq)) + (kill-buffer (tq-buffer tq))) + +(defun tq-filter (tq string) + "Append STRING to the TQ's buffer; then process the new data." + (let ((buffer (tq-buffer tq))) + (when (buffer-live-p buffer) + (with-current-buffer buffer + (goto-char (point-max)) + (insert string) + (tq-process-buffer tq))))) + +(defun tq-process-buffer (tq) + "Check TQ's buffer for the regexp at the head of the queue." + (let ((buffer (tq-buffer tq))) + (when (buffer-live-p buffer) + (set-buffer buffer) + (if (= 0 (buffer-size)) () + (if (tq-queue-empty tq) + (let ((buf (generate-new-buffer "*spurious*"))) + (copy-to-buffer buf (point-min) (point-max)) + (delete-region (point-min) (point)) + (pop-to-buffer buf nil) + (error "Spurious communication from process %s, see buffer %s" + (process-name (tq-process tq)) + (buffer-name buf))) + (goto-char (point-min)) + (if (re-search-forward (tq-queue-head-regexp tq) nil t) + (let ((answer (buffer-substring (point-min) (point)))) + (delete-region (point-min) (point)) + (unwind-protect + (condition-case nil + (funcall (tq-queue-head-fn tq) + (tq-queue-head-closure tq) + answer) + (error nil)) + (tq-queue-pop tq)) + (tq-process-buffer tq)))))))) + +(provide 'tq) + +;;; arch-tag: 65dea08c-4edd-4cde-83a5-e8a15b993b79 +;;; tq.el ends here |